app中数据的共享使用

App Groups
 
这是iOS8新开放的功能,在OS X上早就可用了。它主要用于同一group下的app共享同一份读写空间,以实现数据共享。
 
extension和containing app共同读写一份数据是很合理的需求,比如系统的股市应用,widget和app中都需要展示几个公司的股票数据,这就可以通过App Groups实现。
 
1.1 功能开启
 
为了便于后续操作,请先确保你的开发者账号在Xcode上处于登录状态。
 
在app中开启
App Groups位于:

  1. TARGETS-->AppExtensionDemo-->Capabilities-->App Groups 
找到以后,将App Groups右上角的开关打开,然后选择添加groups,比如我的是group.wangzz,当然这是为了测试随便起得名字,正规点得命名规则应该是:group.com.company.app。
 
添加成功以后如下图所示:
app中数据的共享使用
1.2 在extension中开启
我创建的是widget,target名称为TodayExtension,对应的App Groups位于:

  1. TARGETS-->TodayExtension-->Capabilities-->App Groups 
开启方式和app中一样,需要注意的是必须保证这里地App Groups名称和app中的相同,即为group.wangzz。
 
二、extension和containing app数据共享
 
App Groups给我们提供了同一group内app可以共同读写的区域,可以通过以下方式实现数据共享:
 
2.1 通过NSUserDefaults共享数据
 
存数据
通过以下方式向NSUserDefaults中保存数据:

  1. - (void)saveTextByNSUserDefaults 
  2.     NSUserDefaults *shared = [[NSUserDefaults alloc] initWithSuiteName:@"group.wangzz"]; 
  3.     [shared setObject:_textField.text forKey:@"wangzz"]; 
  4.     [shared synchronize]; 
需要注意的是:
 
1.保存数据的时候必须指明group id;
 
2.而且要注意NSUserDefaults能够处理的数据只能是可plist化的对象,详情见Property List Programming Guide。
 
3.为了防止出现数据同步问题,不要忘记调用[shared synchronize];
 
读数据
对应的读取数据方式:

  1. - (NSString *)readDataFromNSUserDefaults 
  2.     NSUserDefaults *shared = [[NSUserDefaults alloc] initWithSuiteName:@"group.wangzz"]; 
  3.     NSString *value = [shared valueForKey:@"wangzz"]; 
  4.  
  5.     return value; 
 
2.2 通过NSFileManager共享数据
 
NSFileManager在iOS7提供了containerURLForSecurityApplicationGroupIdentifier方法,可以用来实现app group共享数据。
 
保存数据

  1. - (BOOL)saveTextByNSFileManager 
  2.     NSError *err = nil; 
  3.     NSURL *containerURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.wangzz"]; 
  4.     containerURL = [containerURL URLByAppendingPathComponent:@"Library/Caches/good"]; 
  5.  
  6.     NSString *value = _textField.text; 
  7.     BOOL result = [value writeToURL:containerURL atomically:YES encoding:NSUTF8StringEncoding error:&err]; 
  8.     if (!result) { 
  9.         NSLog(@"%@",err); 
  10.     } else { 
  11.         NSLog(@"save value:%@ success.",value); 
  12.     } 
  13.  
  14.     return result; 
 
读数据

  1. - (NSString *)readTextByNSFileManager 
  2.     NSError *err = nil; 
  3.     NSURL *containerURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.wangzz"]; 
  4.     containerURL = [containerURL URLByAppendingPathComponent:@"Library/Caches/good"]; 
  5.     NSString *value = [NSString stringWithContentsOfURL:containerURL encoding:NSUTF8StringEncoding error:&err]; 
  6.  
  7.     return value; 
 
在这里我试着保存和读取的是字符串数据,但读写SQlite我相信也是没问题的。
 
数据同步
两个应用共同读取同一份数据,就会引发数据同步问题。WWDC2014的视频中建议使用NSFileCoordination实现普通文件的读写同步,而数据库可以使用CoreData,Sqlite也支持同步。
 
三、extension和containing app代码共享
 
和数据共享类似,extension和containing app很自然地会有一些业务逻辑上可以共用的代码,这时可以通过iOS8中刚开放使用的framework实现。苹果在App Extension Programming Guide中是这样描述的:
 
In iOS 8.0 and later, you can use an embedded framework to share code between your extension and its containing app. For example, if you develop image-processing code that you want both your Photo Editing extension and its containing app to share, you can put the code into a framework and embed it in both targets.
 
即将framework分别嵌入到extension和containing app的target中实现代码共享。但这样岂不是需要分别要将framework分别copy到extension和containing app的main bundle中?
 
参考extension和containing app数据共享,我试想能不能将framework只保存一份放在App Groups区域?
 
3.1 copy framework到App Groups
 
在app首次启动的时候将framework放到App Groups区域:

  1. - (BOOL)copyFrameworkFromMainBundleToAppGroup 
  2.     NSFileManager *manager = [NSFileManager defaultManager]; 
  3.     NSError *err = nil; 
  4.     NSURL *containerURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.wangzz"]; 
  5.     NSString *sorPath = [NSString stringWithFormat:@"%@/Dylib.framework",[[NSBundle mainBundle] bundlePath]]; 
  6.     NSString *desPath = [NSString stringWithFormat:@"%@/Library/Caches/Dylib.framework",containerURL.path]; 
  7.  
  8.     BOOL removeResult = [manager removeItemAtPath:desPath error:&err]; 
  9.     if (!removeResult) { 
  10.         NSLog(@"%@",err); 
  11.     } else { 
  12.         NSLog(@"remove success."); 
  13.     } 
  14.  
  15.     BOOL copyResult = [[NSFileManager defaultManager] copyItemAtPath:sorPath toPath:desPath error:&err]; 
  16.     if (!copyResult) { 
  17.         NSLog(@"%@",err); 
  18.     } else { 
  19.         NSLog(@"copy success."); 
  20.     } 
  21.  
  22.     return copyResult; 
 
3.2 使用framework:

  1. - (BOOL)loadFrameworkInAppGroup 
  2.     NSError *err = nil; 
  3.     NSURL *containerURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.wangzz"]; 
  4.     NSString *desPath = [NSString stringWithFormat:@"%@/Library/Caches/Dylib.framework",containerURL.path]; 
  5.     NSBundle *bundle = [NSBundle bundleWithPath:desPath]; 
  6.     BOOL result = [bundle loadAndReturnError:&err]; 
  7.     if (result) { 
  8.         Class root = NSClassFromString(@"Person"); 
  9.         if (root) { 
  10.             Person *person = [[root alloc] init]; 
  11.             if (person) { 
  12.                 [person run]; 
  13.             } 
  14.         } 
  15.     } else { 
  16.         NSLog(@"%@",err); 
  17.     } 
  18.  
  19.     return result; 
 
经过测试,竟然能够加载成功。
 
需要说明的是,这里只是说那么用是可以成功加载framework,但还面临不少问题,比如如果用户在启动app之前去使用extension,这时framework还没有copy过去,怎么处理;另外iOS的机制或者苹果的审核是否允许这样使用等。