我们知道iOS由于沙盒的存在,应用程序不能越过自己的区域去访问别的存储空间的内容,不过可能有许多场景我们需要在应用程序之间共享数据,比如多个应用共用用户名密码进行登录等。虽然我们不能直接通过文件系统来分享数据,不过还是有些方法可以实现,为了方便说明,这里同时创建了两个工程Example1和Example2,实现这两个app之间的信息共享,Example1负责写数据,Example2负责读数据,具体的demo代码可以到这里获取
UIPasteboard
剪贴板是应用程序之间传递数据的简单方式,建议不要使用全局的粘贴板,而是自己根据名字创建一个新的粘贴板,防止其它地方全局拷贝的影响。然后把需要共享的内容复制到粘贴板,粘贴板的内容可以是文本、URL、图片和UIColor等,另一个app就可以根据粘贴板的名字去读取相关的信息。
Example1设置粘贴板的内容:
UIPasteboard *pasteboard = [UIPasteboard pasteboardWithName:@"myPasteboard" create:YES];
pasteboard.string = @"myShareData";
Example2读取粘贴板的内容:
UIPasteboard *pasteboard = [UIPasteboard pasteboardWithName:@"myPasteboard" create:NO];
NSString *content = pasteboard.string;
Custom URL Scheme
URL Scheme能够让我们通过自定义的协议在应用程序间传递信息,当你想要发送数据给一个实现了自定义URL Scheme的应用时,只需要创建好合适格式的URL,然后调用openURL:方法,系统就会载入注册了该scheme的应用然后将你的URL传递给他,比如如下的代码,todolist是一个其它应用注册的scheme,通过openURL便可以将服务请求发送到该应用(自定义的URL Scheme要能够唯一标示该APP,如果你的URL Scheme跟别人冲突了,那么你的app就不一定会被调起,iOS并不保证调用哪个应用)。
NSURL *myURL = [NSURL URLWithString:@"todolist://newid=20"];
[[UIApplication sharedApplication] openURL:myURL];
- 注册URL Scheme
为了能够处理URL请求,我们需要先注册自定义的URL Scheme,只需要在Info tab下的URL Types添加即可,比如我们这里注册Example2的URL Scheme。
Identifier用于标示名称,为了唯一性可以采用反转域名的形式,另外我们设置URL Scheme为Example2,以及role为Viewer(Viewer表示只能读取改URL但不能修改,Editor可以对URL进行读写),这样Example2就能够接受类似"Example2:\"的URL请求了,可以在浏览器中输入"Example2:\"链接打开app。
- 处理URL请求
当有URL请求到来时,所有的URL都会传递给你的app delegate,我们需要实现application:openURL:sourceApplication:annotation:方法来接收处理对应的URL,所以假如你想通过URL方式来传递数据,只需要将数据添加到URL中即可,另外的通过解析该URL来获取对应的数据。
Eaxmple1发送携带参数的URL到Example2:
NSString *string = @"Example2://name=jiangbin#age=1";
NSURL *url = [NSURL URLWithString:string];
[[UIApplication sharedApplication] openURL:url];
Example2处理URL请求并解析:
-(BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation
{
if ([[url scheme] isEqualToString:@"Example2"]) {
NSString *content = [url resourceSpecifier];
//解析content获取数据
//...
return YES;
}
return NO;
}
Shared Keychain Access
iOS的keychain提供一种安全保存信息的方式,可以保存密码等数据,而且keychain中的数据不会因为你删除app而丢失,你可以在重新安装后继续读取keychain中的数据。通常每个应用程序只允许访问自己在keychain中保存的数据,不过假如你使用同一个证书的话,不同的app也可以通过keychain来实现应用间的数据共享,之前下载百度贴吧应用的时候发现首次打开它就自动登录了,可能百度的应用之间就是通过这种方式共享用户名密码进行登录的,之前登录过百度云。
为了实现keychain共享数据,我们需要开启Keychain Sharing,开启方法如下,然后添加设置相同的Keychain Group,不过别忘记了添加Security.framework。
Example1保存数据到keychain(为了简单使用SSKeychian)
- (void)setKeyChain
{
[SSKeychain setPassword:@"shareData" forService:@"myservice" account:@"jiangbin"];
}
Example2读取数据
- (IBAction)getByKeychain:(id)sender
{
NSString *myData = [SSKeychain passwordForService:@"myservice" account:@"jiangbin"];
}
App Groups
iOS8之后苹果加入了App Groups功能,应用程序之间可以通过同一个group来共享资源,app group可以通过NSUserDefaults进行小量数据的共享,如果需要共享较大的文件可以通过NSFileCoordinator、NSFilePresenter等方式。
开启app groups,需要添加一个group name,app之间通过这个group共享数据:
Example1根据group name设置内容:
- (void)setAppGroup
{
NSUserDefaults *myDefaults = [[NSUserDefaults alloc]
initWithSuiteName:@"group.com.jiangbin.SharedData"];
[myDefaults setObject:@"shared data" forKey:@"mykey"];
}
Example2根据group name读取数据
- (void)getByAppGroup
{
NSUserDefaults *myDefaults = [[NSUserDefaults alloc]
initWithSuiteName:@"group.com.jiangbin.SharedData"];
NSString *content = [myDefaults objectForKey:@"mykey"];
}