App Extensions 是iOS8新开放的扩展机制,之后不断增加功能。App Extension Programming Guide: Today
 
不喜欢废话,直接上干货!
 
一:重要概念:
extension point
系统中支持extension的区域,extension的类别也是据此区分的,iOS上共有Today、Share、Action、Photo Editing、Storage Provider、Custom keyboard等等种类,其中Today中的extension又被称为widget。每种extension point的使用方式和适合干的活都不一样,因此不存在通用的extension。
 
app extension
app extension并不是一个独立的app,它有一个包含在主app bundle中的独立bundle,extension的bundle后缀名是.appex。其生命周期也和普通app不同。extension不能单独存在,必须有一个包含它的containing app。另外,extension需要用户手动激活,不同的extension激活方式也不同,比如: 比如Today中的widget需要在Today中激活和关闭。
 
containing app
尽管苹果开放了extension,但是在iOS中extension并不能单独存在,要想提交到AppStore,必须将extension包含在一个app中提交,并且app的实现部分不能为空,这个包含extension的app就叫containing app。extension会随着containing app的安装而安装,同时随着
containing app的卸载而卸载。
 
host app
能够调起extension的app被称为host app,比如widget的host app就是Today。
 
二、extension和containing app、host app
 
extension和host app
extension和host app之间可以通过extensionContext属性直接通信,该属性是新增加的UIViewController分类:

实际上extension和host app之间是通过IPC(interprocess communication)实现的,只是苹果把调用接口高度抽象了,我们并不需要关注那么底层的东西。

 
containing app和host app
他们之间没有任何直接关系,也从来不需要通信。
 
extension和containing app
不能直接通信
首先,尽管extension的bundle是放在containing app的bundle中,但是他们是两个完全独立的进程,之间不能直接通信。不过extension可以通过openURL的方式启动containing app(当然也能启动其它app),不过必须通过extensionContext借助host app来实现.
 
extension中是无法直接使用openURL的,所以使用  self.extensionContext 。
另外:
 
然后再主APP上面进行截取,判断,可以定点打开某个原生页面,或者H5页面
 
可以共享Shared resources
 
extension和containing app可以共同读写一个被称为Shared resources的存储区域,这是通过App Groups实现的,后文将会详述。
 
containing app能够控制extension的出现和隐藏
通过以下代码,containing app可以让extension出现或隐藏(当然extension也可以让自己隐藏):
  1. //让隐藏的插件重新显示
  2. - (void)showTodayExtension
  3. {
  4. [[NCWidgetController widgetController] setHasContent:YES forWidgetWithBundleIdentifier:@"com.yoowei.app.extension"];
  5. }
  6. //隐藏插件
  7. - (void)hiddeTodayExtension
  8. {
  9. [[NCWidgetController widgetController] setHasContent:NO forWidgetWithBundleIdentifier:@"com.yoowei.app.extension"];
  10. }
 
三、App Groups
 
这是iOS8新开放的功能 , 它主要用于同一group下的app共享同一份读写空间,以实现数据共享。
extension和containing app共同读写一份数据是很合理的需求.
 
3.1 功能开启
 
为了便于后续操作,请先确保你的开发者账号在Xcode上处于登录状态。
 
在containing app中开启
App Groups位于:
  1. TARGETS-->AppExtensionYoowei-->Capabilities-->App Groups
找到以后,将App Groups右上角的开关打开,然后选择添加groups,比如我的是group.com.yoowei.app (可以随便起)
 
在extension中开启
我创建的是widget,target名称为TodayExtension,对应的App Groups位于:
  1. TARGETS-->TodayExtension-->Capabilities-->App Groups
开启方式和app中一样,需要注意的是必须保证这里地App Groups名称和app中的相同,即为group.com.yoowei.app。
 
Certificates, Identifiers & Profilest添加
Identifiers -- App Groups  选择右上角➕号添加即可,然后还可以编辑。
 
四、extension和containing app数据共享
 
App Groups给我们提供了同一group内app可以共同读写的区域,可以通过以下方式实现数据共享:
 
4.1 通过NSUserDefaults共享数据
 
存数据
通过以下方式向NSUserDefaults中保存数据:
  1. - (void)saveTextByNSUserDefaults
  2. {
  3. NSUserDefaults *shared = [[NSUserDefaults alloc] initWithSuiteName:@"group.com.yoowei.app"];
  4. [shared setObject:_textField.text forKey:@"yoowei"];
  5. [shared synchronize];
  6. }
需要注意的是:
 
1.保存数据的时候必须指明group id;
 
2.而且要注意NSUserDefaults能够处理的数据只能是可plist化的对象,详情见Property List Programming Guide。
 
3.为了防止出现数据同步问题,不要忘记调用[shared synchronize];
 
读数据
对应的读取数据方式:
  1. - (NSString *)readDataFromNSUserDefaults
  2. {
  3. NSUserDefaults *shared = [[NSUserDefaults alloc] initWithSuiteName:@"group.com.yoowei.app"];
  4. NSString *value = [shared valueForKey:@"yoowei"];
  5. return value;
  6. }
 
4.2 通过NSFileManager共享数据
 
NSFileManager在iOS7提供了containerURLForSecurityApplicationGroupIdentifier方法,可以用来实现app group共享数据。
 
保存数据
  1. - (BOOL)saveTextByNSFileManager
  2. {
  3. NSError *err = nil;
  4. NSURL *containerURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.com.yoowei.app"];
  5. containerURL = [containerURL URLByAppendingPathComponent:@"Library/Caches/good"];
  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. return result;
  14. }
 
读数据
  1. - (NSString *)readTextByNSFileManager
  2. {
  3. NSError *err = nil;
  4. NSURL *containerURL = [[NSFileManager defaultManager]  containerURLForSecurityApplicationGroupIdentifier:@"group.com.yoowei.app"];
  5. containerURL = [containerURL URLByAppendingPathComponent:@"Library/Caches/good"];
  6. NSString *value = [NSString  stringWithContentsOfURL:containerURL encoding:NSUTF8StringEncoding error:&err];
  7. return value;
  8. }
在这里我试着保存和读取的是字符串数据,但读写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区域?
 
5.1 copy framework到App Groups
 
在app首次启动的时候将framework放到App Groups区域:
  1. - (BOOL)copyFrameworkFromMainBundleToAppGroup
  2. {
  3. NSFileManager *manager = [NSFileManager defaultManager];
  4. NSError *err = nil;
  5. NSURL *containerURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.com.yoowei.app"];
  6. NSString *sorPath = [NSString stringWithFormat:@"%@/Dylib.framework",[[NSBundle mainBundle] bundlePath]];
  7. NSString *desPath = [NSString stringWithFormat:@"%@/Library/Caches/Dylib.framework",containerURL.path];
  8. BOOL removeResult = [manager removeItemAtPath:desPath error:&err];
  9. if (!removeResult) {
  10. NSLog(@"%@",err);
  11. } else {
  12. NSLog(@"remove success.");
  13. }
  14. BOOL copyResult = [[NSFileManager defaultManager] copyItemAtPath:sorPath toPath:desPath error:&err];
  15. if (!copyResult) {
  16. NSLog(@"%@",err);
  17. } else {
  18. NSLog(@"copy success.");
  19. }
  20. return copyResult;
  21. }
 
5.2 使用framework:
  1. - (BOOL)loadFrameworkInAppGroup
  2. {
  3. NSError *err = nil;
  4. NSURL *containerURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.com.yoowei.app"];
  5. NSString *desPath = [NSString stringWithFormat:@"%@/Library/Caches/Dylib.framework",containerURL.path];
  6. NSBundle *bundle = [NSBundle bundleWithPath:desPath];
  7. BOOL result = [bundle loadAndReturnError:&err];
  8. if (result) {
  9. Class root = NSClassFromString(@"Person");
  10. if (root) {
  11. Person *person = [[root alloc] init];
  12. if (person) {
  13. [person run];
  14. }
  15. }
  16. } else {
  17. NSLog(@"%@",err);
  18. }
  19. return result;
  20. }
 
经过测试,竟然能够加载成功。
 
需要说明的是,这里只是说那么用是可以成功加载framework,但还面临不少问题,比如如果用户在启动app之前去使用extension,这时framework还没有copy过去,怎么处理;另外iOS的机制或者苹果的审核是否允许这样使用等。
 
在一切确定下来之前还是乖乖按文档中的方式使用吧。
 
六、生命周期
 
extension和普通app的最大区别之一是生命周期。
 
开始
在用户通过host app点击extension时,系统就会实例化extension应用,这是生命周期的开始。
 
执行任务
在extension启动以后,开始执行它的使命。
 
终止
在用户取消任务,或者任务执行结束,或者开启了一个长时后台任务时,系统会将其杀掉。
 
由此可见,extension就是为了任务而生!
 
通过打印日志发现,Today中的widget在将Today切换到全部或者未读通知时都会被杀掉。
 
七、 调试
 
extension和普通app的调试方式差不多,开始调试前先选中extension对应的scheme,点击run.不出意外的话,会弹出一个框:
choose an app to run  也就是说需要选择一个host app,这里选择Today。
 
注意:
开始调试前先选中extension对应的scheme。
 
八、 iOS8应用文件系统
 
发现iOS8的文件系统发生了变化,新的文件系统将可执行文件(即原来的.app文件)从沙盒中移到了另外一个地方,这样感觉更合理。
 
测试代码
下述代码用于打印App Groups路径、应用的可执行文件路径、对应的Documents路径:
  1. - (void)logAppPath
  2. {
  3. //app group路径
  4. NSURL *containerURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.com.yoowei.app"];
  5. NSLog(@"app group:\n%@",containerURL.path);
  6. //打印可执行文件路径
  7. NSLog(@"bundle:\n%@",[[NSBundle mainBundle] bundlePath]);
  8. //打印documents
  9. NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
  10. NSString *path = [paths objectAtIndex:0];
  11. NSLog(@"documents:\n%@",path);
  12. }
不管是extension还是containing app,他们的可执行文件和保存数据的目录都是分开存放的,即所有app的可执行文件都放在一个大目录下,保存数据的目录保存在另一个大目录下,同样,AppGroup放在另一个大目录下。
 
 
参考文档
 
http://www.cocoachina.com/industry/20140627/8960.html
 
 
 
 
 
 
 
 
 

App Extension Today的更多相关文章

  1. App Extension访问Cocoapods引入的第三方库

    步骤一: PROJECT --info --configurations,将对应的Debug和Release 设置成pods.debug和pods.release    步骤2:编译一下(本人遇到的问 ...

  2. 关于报错:'sharedApplication' is unavailable: not available on iOS (App Extension) - Use view controller based

    最近在看Extension相关知识的时候,自己写了个小demo 发现[UIApplication sharedApplication]这个方法敲不出来了, 总是报错:'sharedApplicatio ...

  3. App Extension编程指南(iOS8/OS X v10.10)中文版

    http://www.cocoachina.com/ios/20141023/10027.html 当iOS 8.0和OS X v10.10发布后,一个全新的概念出现在我们眼前,那就是应用扩展.顾名思 ...

  4. iOS9中找不到XXX.dylib 与 is unavailable no availabel on ios (app extension) - use view controller 的解决办法

    在 iOS9 中现在找不到 XXX.dylib 了,比如libz.tbd  如果要用到 libz.dylib,可以用下面的办法,来自 Stack Overflow. Go to Build Phase ...

  5. iOS - App Extension 整体总结

    一.App Extension的介绍 App Extension可以让你扩展你APP的自定义功能和内容,使用户可以在与其他应用或者系统进行互动的时候去使用它.app extension即为本文所说的e ...

  6. App Extension的脱壳办法

    App Extension的脱壳办法 从app store下载的app和app extension是加过密的,可以通过otool查看: $ otool -l binary_name | grep cr ...

  7. iOS 在Host App 与 App Extension 之间发送通知

    如何从你的一个App发送通知给另一个App? (例:搜狗输入法下载皮肤完成后使用皮肤) 注:搜狗输入法是App.而键盘是Extension 当你为你的App 添加 App Extension时,如果想 ...

  8. 揭秘 iOS App Extension 开发 —— Today 篇

    转自:http://www.cocoachina.com/ios/20160619/16760.html 本文授权转载,作者:Cyandev(简书) 从 iOS 8 开始,苹果引入了全新的 App E ...

  9. iOS App Extension入门

    转自简书:http://www.jianshu.com/p/8cf08db29356   iOS 10推出了很多新功能,其中有几个高调的变化:通知栏更加实用,电话可以防骚扰,iMessage变得更加有 ...

随机推荐

  1. 内存溢出OOM与内存泄漏ML

    附, 微信团队原创分享:Android内存泄漏监控和优化技巧总结 一.如何避免OOM 异常 想要避免OOM 异常首先我们要知道什么情况下会导致OOM 异常. 1.图片过大导致OOM Android 中 ...

  2. HTML常用标签与表格标签

    超链接标签: <a href="超链接地址" target="_blank">超链接的文字</a> _blank或new是在新网页中打开 ...

  3. WordPress菜单函数wp_nav_menu()详细介绍

    导航菜单函数wp_nav_menu()进行详细的说明. 1.wp_nav_menu()函数介绍: worpdress发展到3.0以后增加了一个自定义菜单函数wp_nav_menu(),使得wordpr ...

  4. UML类图中的六种关系及实例

    前言: 设计模式是一种对于面向对象语言(C#,C++,Java)的高级应用.其思维体现出的是真正的代码设计.每一种模式都堪称巧妙!但基于各种设计模式,这里少不了基本的类图设计,本文简要列出6种关系,及 ...

  5. C语言与套接字

    我们已经知道如何使用I/O与文件通信,还知道了如何让同一计算机上的两个进程进行通信,这篇文章将创建具有服务器和客户端功能的程序 互联网中大部分的底层网络代码都是用C语言写的. 网络程序通常有两部分组成 ...

  6. vim快捷键总结

    直接上图 原图地址:vim快捷键

  7. 如何使用maven建一个web3.0的项目

    使用eclipse手动建一个maven的web project可能会有版本不合适的情况,例如使用spring的websocket需要web3.0什么的,不全面的修改可能会出现各种红叉,甚是苦恼.我从我 ...

  8. Redis有序集合Zset(sorted set)

    zadd/zrange 127.0.0.1:6379> zadd zset01 60 v1 70 v2 80 v3 90 v4 100 v5(integer) 5127.0.0.1:6379&g ...

  9. spring的自动装配基础

    当开始看别人的代码使用注解的时候,以为照着别人的代码写,也写一个注释就能实现这样的功能,但是,现在开始考虑自动装配时怎样实现的. 首先,如果如果知道如何手动在xml配置中"装配bean&qu ...

  10. UOJ262 【NOIP2016】换教室

    本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000作者博客:http://www.cnblogs.com/ljh2000-jump/转 ...