iCloud文档存储编程相对键值数据存储而言比较复杂,涉及到自定义文档类、获得iCloud文档目录、查找Ubiquity容器中的文档、保存文档和解决文档冲突等内容。

 

实例:iCloud文档存储编程设计

实例介绍一下iCloud文档存储编程过程,画面中有一个文本框和一个按钮控件,在设备1输入内容,点击“保存数据”按钮,将数据保存iCloud服务器。右图是设备2画面,过几秒钟后设备2上会读取iCloud服务器端数据,并显示在文本框中。

配置Xcode工程

编写iCloud文档存储编应用程序也需要在Xcode工程中进行一些配置,选择TAGETS→MyNotes→Summary→Entitlements

15

在图中Ubiquity Contrainers添加com.51work6.MyNotes,这是Ubiquity容器标识,可以有多个Ubiquity容器标识,这个容器标识代表着不同的目录。com.51work6.MyNotes代表目录如下所示:

/var/mobile/Library/Mobile Documents/98Z3R5XU29~com~51work6~MyNotes/

其中的98Z3R5XU29是在iOS开发者配置门户网站创建App ID时候生成的,它被称为Team ID

如果应用中没有使用iCloud键值数据存储key-Value Store可以不用配置。

自定义文档类

我 们需要自己封装一个文档类,它继承抽象类UIDocument,而UIDocument实现NSFilePresenter协议。实现 NSFilePresenter协议的类,它所代表的文件和目录可以被查看和编辑,这些NSFilePresenter实现类与文件协调者类 NSFileCoordinator结合使用,可以协调管理文件或目录。

这个类我们命名为MyCloudDocument, MyCloudDocument的代码如下:

 

   
   
   
   
  1. #import  
  2.  
  3.   
  4.  
  5. @interface MyCloudDocument : UIDocument ① 
  6.  
  7. @property (strong, nonatomic) NSString *contents; 
  8.  
  9. @end 
  10.  
  11.   
  12.  
  13. #import ”MyCloudDocument.h” 
  14.  
  15. @implementation MyCloudDocument 
  16.  
  17.   
  18.  
  19. //加载数据 
  20.  
  21. - (BOOL)loadFromContents:(id)contents ofType:(NSString *)typeName error:(NSError **)outError ② 
  22.  
  23.  
  24. if ([contents length] > 0) 
  25.  
  26.  
  27. self.contents = [[NSString alloc] initWithData:contents encoding:NSUTF8StringEncoding]; ③ 
  28.  
  29.  
  30. return YES; 
  31.  
  32.  
  33. //保存数据 
  34.  
  35. - (id)contentsForType:(NSString *)typeName error:(NSError **)outError ④ 
  36.  
  37.  
  38. return [self.contents dataUsingEncoding:NSUTF8StringEncoding];  ⑤ 
  39.  
  40.  
  41. @end 

获得iCloud文档目录

iCloud文档目录是指在Ubiquity容器下的Document目录,因此获得了Ubiquity容器的根目录,就可以获得的iCloud文档目录了。

ViewController.m中的 ubiquitousDocumentsURL方法可以iCloud文档目录:

 

   
   
   
   
  1. //请求本地Ubiquity容器,从容器中获得Document目录URL 
  2.  
  3. - (NSURL *)ubiquitousDocumentsURL { 
  4.  
  5. NSFileManager* fileManager = [NSFileManager defaultManager]; ① 
  6.  
  7. NSURL* containerURL = [fileManager 
  8.  
  9. URLForUbiquityContainerIdentifier:@"98Z3R5XU29.com.51work6.MyNotes"]; ② 
  10.  
  11. containerURL = [containerURL URLByAppendingPathComponent:@"Documents"]; ③ 
  12.  
  13. return containerURL; 
  14.  

查找Ubiquity容器中的文档

获得iCloud文档目录后,我们需要找到容器中的文件。查询容器中的文件需要注册两个广播通知:

NSMetadataQueryDidFinishGatheringNotification  查询结束发出通知;

NSMetadataQueryDidUpdateNotification  查询结束,进入开始更新阶段发出的通知;

ViewController.m中注册和解除通知代码如下:

 

   
   
   
   
  1. - (void)viewDidLoad 
  2.  
  3.  
  4. [super viewDidLoad]; 
  5.  
  6. //为查询iCloud文件的变化,注册通知 
  7.  
  8. [[NSNotificationCenter defaultCenter] addObserver:self 
  9.  
  10. selector:@selector(updateUbiquitousDocuments:) 
  11.  
  12. name:NSMetadataQueryDidFinishGatheringNotification object:nil]; 
  13.  
  14. [[NSNotificationCenter defaultCenter] addObserver:self 
  15.  
  16. selector:@selector(updateUbiquitousDocuments:) 
  17.  
  18. name:NSMetadataQueryDidUpdateNotification object:nil]; 
  19.  
  20.   
  21.  
  22. //查询iCloud文件的变化 
  23.  
  24. [self searchFilesOniCloud]; 
  25.  
  26.   
  27.  
  28.  
  29. - (void)didReceiveMemoryWarning 
  30.  
  31.  
  32. [super didReceiveMemoryWarning]; 
  33.  
  34. [[NSNotificationCenter defaultCenter] removeObserver:self]; 
  35.  
  36.  
  37.   

保存文档 

保存文档很简单,它是在ViewController.m中的 saveClick:方法处理的,saveClick:方法是点击按钮时候触发:

 

   
   
   
   
  1. - (IBAction)saveClick:(id)sender { 
  2.  
  3. _myCloudDocument.contents = _txtContent.text; 
  4.  
  5. [_myCloudDocument updateChangeCount:UIDocumentChangeDone]; 
  6.  
  7. [_txtContent resignFirstResponder]; 
  8.  

保 存成功之后我们可以在其它设备上看看是否iCloud中已经有abc.txt文档了。如果使用Mac OS X系统电脑,我们可以在“系统偏好设置”中打 开iCloud对话框,点击“管理”按钮可以打开iCloud空间管理对话框,其中的MyNotes是我应用名,右边的abc.txt创建的文件。

如果在iPhone、iPod touch和iPad等iOS设备中查看,可以启动设置应用程序,也进入到存储空间管理中,我的iPod touch中查看的情况。

17

出自《iOS网络编程与云端应用最佳实践》作者:关东升 @tony_关东升