1、前言
iOS文件存储机制
每个iOS程序都有一个独立的文件系统(存储空间),而且只能在对应的文件系统中进行操作,此区域被称为沙盒。应用必须待在自己的沙盒里,其他应用不能访问该沙盒。所有的非代码文件都要保存在此,例如属性文件plist、文本文件、图像、图标、媒体资源等
沙盒中相关路径
AppName.app 应用程序的程序包目录,包含应用程序的本身。由于应用程序必须经过签名,所以不能在运行时对这个目录中的内容进行修改,否则会导致应用程序无法启动。
Documents/ 保存应用程序的重要数据文件和用户数据文件等。用户数据基本上都放在这个位置(例如从网上下载的图片或音乐文件),该文件夹在应用程序更新时会自动备份,在连接iTunes时也可以自动同步备份其中的数据。
Library:这个目录下有两个子目录,可创建子文件夹。可以用来放置您希望被备份但不希望被用户看到的数据。该路径下的文件夹,除Caches以外,都会被iTunes备份.
Library/Caches: 保存应用程序使用时产生的支持文件和缓存文件(保存应用程序再次启动过程中需要的信息),还有日志文件最好也放在这个目录。iTunes 同步时不会备份该目录并且可能被其他工具清理掉其中的数据。
Library/Preferences: 保存应用程序的偏好设置文件。NSUserDefaults类创建的数据和plist文件都放在这里。会被iTunes备份。
相关问题
因为每个app下的沙盒路径都是封闭的,其他应用访问不到,也就导致了文件不能共享。只能自己在代码中维护本app内的文件。这给用户带来很多的不便。因此,苹果除了一个系统的“文件”的app,用来管理app内的文件。但是系统的“文件”这个app不是可以管理全部的app下的文件,他只能管理对“文件”这个app开放权限的APP。
计划
最近在做IM开发,涉及到文件传输,我们不想做的特别封闭,所以就想可以发送app以外的文件(比如微信中保存下来的)具体实现就是:从微信里下载的文件保存到“ fileAPP”内,然后我们在发送文件时,可以发送本地文件,也可以发送“ fileAPP”内保存的文件
2、让自己的app对“ fileAPP”开放管理权限
2.1、在Identifiers下选择你要添加icloud的boundid把icloud配置勾选上
2.2、工程配置
点击Caoablity左侧的加号,搜索iCloud,然后添加即可
2.3、 设置info.plist
第一个是 UIFileSharingEnabled,这个可以使 iTunes 分享你文件夹内的内容;第二个是 LSSupportsOpeningDocumentsInPlace ,它保证了你文件夹内本地文件的获取权限,你需要将这两个键值对的值设置为 YES
以上设置完,重新运行app之后,在系统的“ fileAPP”中会出现一个以你的APP命名的文件夹,里面包含了APP内沙盒的目录和文件
但是这里面的文件很乱,而且会将沙盒内 Documents 文件夹内的所有文件都显示出来
2.4、如何在fileAPP里隐藏重要的文件?
如果它不是那么重要的,我们可以将它们存放在 cachesDirectory 或者是 temporaryDirectory 文件夹下面;如果它是重要的文件,大多数情况下,我们是需要将它们备份在 iCloud 上的,这样的文件我们建议将它存放在 applicationSupportDirectory 目录下
以上,是将自己的文件共享给 fileAPP
3、app如何获取到“文件”app下管理的文件到本app内
3.1、调起fileAPP 文件目录
//打开文件APP
- (void)presentDocumentCloud {
NSArray *documentTypes = @[@"public.content", @"public.text", @"public.source-code ", @"public.image", @"public.audiovisual-content", @"com.adobe.pdf", @"com.apple.keynote.key", @"com.microsoft.word.doc", @"com.microsoft.excel.xls", @"com.microsoft.powerpoint.ppt"];
UIDocumentPickerViewController *documentPickerViewController = [[UIDocumentPickerViewController alloc] initWithDocumentTypes:documentTypes inMode:UIDocumentPickerModeOpen];
documentPickerViewController.delegate = self;
[self presentViewController:documentPickerViewController animated:YES completion:nil];
}
这里的documentTypes数组内设置要拿的文件格式
UIDocumentPickerMode有四种:
typedef NS_ENUM(NSUInteger, UIDocumentPickerMode) {
UIDocumentPickerModeImport,
UIDocumentPickerModeOpen,
UIDocumentPickerModeExportToService,
UIDocumentPickerModeMoveToService
} API_DEPRECATED("Use appropriate initializers instead",ios(8.0,14.0)) API_UNAVAILABLE(tvOS);
UIDocumentPickerModeImport : 将文件拿出来UIDocumentPickerModeOpen:打开文件
后面是将文件传到fileAPP内的操作,类似于微信的保存文件到fileAPP内的操作。这里官方文档之给了前两个,后面的我们也没涉及到,以后研究吧。
3.2、设置代理
<UIDocumentPickerDelegate, UIDocumentInteractionControllerDelegate>
3.3、实现代理
#pragma mark - UIDocumentPickerDelegate
- (void)documentPicker:(UIDocumentPickerViewController *)controller didPickDocumentAtURL:(NSURL *)url {
BOOL fileUrlAuthozied = [url startAccessingSecurityScopedResource];
if (fileUrlAuthozied && [self iCloudEnable]) {
//通过文件协调工具来得到新的文件地址,以此得到文件保护功能
NSFileCoordinator *fileCoordinator = [[NSFileCoordinator alloc] init];
NSError *error;
[fileCoordinator coordinateReadingItemAtURL:url options:0 error:&error byAccessor:^(NSURL *newURL) {
//读取文件
NSString *fileName = [newURL lastPathComponent];
fileName = [fileName stringByRemovingPercentEncoding];
NSData * data = [NSData dataWithContentsOfURL:newURL];
self.completionBlock(data,fileName);
self.completionBlock = nil;
}];
[url stopAccessingSecurityScopedResource];
}
}
这里需要将拿到的url做处理,应为这里拿到的url是fileAPP文件的本地文件路径,在自己的app内通过[NSData dataWithContentsOfURL:url]方法是无法拿到数据的,应该是只能拿自己沙盒文件路径的设置的原因。
[fileCoordinator coordinateReadingItemAtURL:url options:0 error:&error byAccessor:^(NSURL *newURL) {
}];
这么处理后,拿到的newUrl就是本地的路径了,通过[NSData dataWithContentsOfURL:url]即可拿到文件data。
之后即可对文件进行发送上传等操作。