1.什么是Extension?
擴(kuò)展 (Extension) 是 iOS 8 和 OSX 10.10 加入的一個(gè)非常大的功能點(diǎn),開(kāi)發(fā)者可以通過(guò)系統(tǒng)提供給我們的擴(kuò)展接入點(diǎn) (Extension point) 來(lái)為系統(tǒng)特定的服務(wù)提供某些附加的功能。
例如1:安裝了同花順應(yīng)用后,打開(kāi)手機(jī)的Widget界面,我們可以看到同花順提供的自選股實(shí)時(shí)行情,這個(gè)功能就是一個(gè)Extension,稱(chēng)為T(mén)oday Extension。
例如2: 安裝了搜狗輸入法之后,在手機(jī)-設(shè)置-鍵盤(pán)-添加新鍵盤(pán)后,就可以用搜狗輸入法替代原生輸入法,這是Custom Keyboard Extension。
目前Xcode8.1提供的所有的擴(kuò)展類(lèi)型如下:

2.Extension 存在的形式
- 創(chuàng)建一個(gè)Extension之前,需要有一個(gè)承載擴(kuò)展的應(yīng)用,稱(chēng)為containing app,然后在Containing app中通過(guò)Xcode->File->new->target->選擇所需的Extension創(chuàng)建。
- Extension 不能單獨(dú)存在和發(fā)布,隨containing app 的安裝而安裝,隨containing app的發(fā)布而發(fā)布,一個(gè)containing app 可以添加多個(gè)Extension。
- Extension 的運(yùn)行是獨(dú)立的,containing app 在沒(méi)有啟動(dòng)的狀態(tài)下,Extension仍然可以被啟動(dòng)和運(yùn)行。
- Extension 可以被系統(tǒng)直接調(diào)用,例如下拉通知欄查看同花順行情時(shí),就是由通知中心啟動(dòng)調(diào)用同花順提供的extension。Extension 也可以被其它應(yīng)用間接調(diào)用,例如在某個(gè)應(yīng)用中調(diào)用搜狗輸入法。
- Extension 有自己的Bundle Identifier,需要在開(kāi)發(fā)者賬號(hào)中注冊(cè)App id和創(chuàng)建Provisioning profile。Extension 的 Bundle Identifier 必須以containing app的 Bundle Identifier為前綴。
例如:containing app 的 Bundle Identifier 為com.abc.app1
Extension app 的 Bundle Identifier 必須為 com.abc.app1.xxx
其中xxx為自定義部分
3. Extension 生命周期詳解
Extension 不是一個(gè)app,所以生命周期和運(yùn)行環(huán)境和app不同。在多數(shù)情況下,extension是由用戶(hù)在某一個(gè)app的界面或者某一個(gè)活動(dòng)的控制器中啟動(dòng),這個(gè)啟動(dòng)extension的app被稱(chēng)為host app。Host app 提供extension運(yùn)行所需的上下文并通過(guò)發(fā)送一個(gè)request的方式開(kāi)啟extension的生命周期,extension在完成host app請(qǐng)求的任務(wù)后結(jié)束運(yùn)行。

蘋(píng)果開(kāi)發(fā)者文檔中原文如下:
Because an app extension is not an app, its life cycle and environment are different. In most cases, an extension launches when a user chooses it from an app’s UI or from a presented activity view controller. An app that a user employs to choose an app extension is called a host app. A host app defines the context provided to the extension and kicks off the extension life cycle when it sends a request in response to a user action. An extension typically terminates soon after it completes the request it received from the host app.
4.Extension的通信
- 一般Extension 主要跟host app 進(jìn)行通信,兩者可以通過(guò)extensionContext 直接通信
@interface UIViewController(NSExtensionAdditions)
// Returns the extension context. Also acts as a convenience method for a view controller to check if it participating in an extension request.
@property (nonatomic,readonly,retain) NSExtensionContext *extensionContext NS_AVAILABLE_IOS(8_0);
@end
- Extension 跟containing app 一般不進(jìn)行通信。甚至extension在運(yùn)行的時(shí)候,containing app 都沒(méi)有啟動(dòng)和運(yùn)行。extension 無(wú)法直接讀取containing app 的沙盒數(shù)據(jù)。
- 只有 Today Extension 可以通過(guò)類(lèi)目NSExtensionContext 中提供的openURL:completionHandler:的方式打開(kāi)containing app,containing app 不可以通過(guò)openURL的方式調(diào)用extension。
A Today widget (and no other app extension type) can ask the system to open its containing app by calling the openURL:completionHandler: method of the NSExtensionContext class.
- Extension 和containing app 可以通過(guò)共同讀寫(xiě)一個(gè)被稱(chēng)為Shared resources的存儲(chǔ)區(qū)域共享本地存儲(chǔ)數(shù)據(jù),這是通過(guò)App Groups實(shí)現(xiàn)的。
