這是一個(gè)模塊化的框架儡蔓,引入了中間層BeeHive來(lái)注冊(cè)和持有各個(gè)模塊和服務(wù)外厂,模塊間的調(diào)用通過(guò)protocol來(lái)解耦泥畅,但是同時(shí)每個(gè)模塊都需要依賴protocol。每個(gè)模塊中的moduleClass和services并沒(méi)有很強(qiáng)的依賴凭豪,moduleClass負(fù)責(zé)一些系統(tǒng)焙蹭,通用事件的回調(diào),service負(fù)責(zé)業(yè)務(wù)和邏輯的實(shí)現(xiàn)嫂伞。
1.BeeHive的框架結(jié)構(gòu)
這個(gè)架構(gòu)圖可以這樣理解:
1.Context是一個(gè)上下文孔厉,保存一些系統(tǒng)配置,需要注冊(cè)的module和service列表帖努,app的編譯環(huán)境等等撰豺。它可以和BHCore以及模塊交互。
2.BHCore是BeeHive的核心代碼拼余,負(fù)責(zé)module和service的注冊(cè)污桦,模塊之間的互調(diào)(其實(shí)也就是模塊里service的調(diào)度)。
3.module包括一個(gè)moduleClass和多個(gè)serviceClass匙监。
簡(jiǎn)化一下就變成下圖凡橱,更易于理解:
2.BeeHive需要了解的幾個(gè)要點(diǎn)
1.模塊指的是什么?
這里的模塊是一個(gè)整體概念亭姥,其中包括一個(gè)moduleClass和多個(gè)service稼钩。
2.模塊注冊(cè)和服務(wù)注冊(cè)是什么意思?
模塊注冊(cè)可以理解為為整個(gè)模塊注冊(cè)了一個(gè)可以接受事件(系統(tǒng)事件达罗,通用事件以及業(yè)務(wù)自定義事件)的類(lèi)坝撑,其實(shí)就是moduleClass的實(shí)例化。
3.事件指的是什么氮块?如何定義業(yè)務(wù)事件绍载?事件在哪里注冊(cè)?
事件包括系統(tǒng)事件滔蝉,通用事件和業(yè)務(wù)自定義事件。系統(tǒng)事件塔沃,指的是Application的生命周期事件蝠引,比如:EnterBackground,EnterForeground蛀柴。通用事件螃概,是BeeHive自己定義的模塊生命事件,比如:moduleSetup鸽疾,moduleInit吊洼。業(yè)務(wù)自定義事件,需要用戶自己擴(kuò)展制肮,繼承BHAppdelegate實(shí)現(xiàn)相應(yīng)事件的注冊(cè)冒窍。BeeHive自己定義了一個(gè)BHAppdelegate递沪,入口Appdelegate需要繼承BHAppdelegate,所有的事件也是在Appdelegate里定義和調(diào)用综液。
4.moduleClass和service之間的關(guān)系款慨?
moduleClass和service其實(shí)是一個(gè)模塊之間的不同組成部分,它們之間并沒(méi)有很強(qiáng)的邏輯關(guān)系谬莹,moduleClass負(fù)責(zé)接收各種事件以及事件的處理檩奠,而service負(fù)責(zé)業(yè)務(wù)的功能實(shí)現(xiàn)。比如說(shuō)附帽,可以在moduleClass生命周期函數(shù)注冊(cè)service埠戳。模塊之間的通訊需要通過(guò)service暴露出來(lái)的協(xié)議接口來(lái)實(shí)現(xiàn)。
5.BeeHive怎么模塊解耦蕉扮?
這個(gè)問(wèn)題主要是框架設(shè)計(jì)的問(wèn)題乞而,首先業(yè)務(wù)代碼之間不互相直接依賴,通過(guò)BeeHive來(lái)獲取相應(yīng)的實(shí)例慢显,然后通過(guò)全局protocol暴露出來(lái)的接口進(jìn)行函數(shù)調(diào)度爪模。說(shuō)白了,就是通過(guò)協(xié)議解耦荚藻,但同時(shí)所有的service都會(huì)依賴這個(gè)protocol屋灌,類(lèi)似于隔了一層中間層。
6.模塊注冊(cè)方式应狱?
(1)靜態(tài)注冊(cè)
也就是本地加載共郭,通過(guò)bundle加載模塊生成數(shù)據(jù),如下圖:
(2)動(dòng)態(tài)注冊(cè)
#define BH_EXPORT_MODULE(isAsync) \
+ (void)load { [BeeHive registerDynamicModule:[self class]]; } \
-(BOOL)async { return [[NSString stringWithUTF8String:#isAsync] boolValue];}
@interface MessageModule ()<BHModuleProtocol>
@end
@implementation MessageModule
BH_EXPORT_MODULE();
@end
調(diào)用BH_EXPORT_MODULE()宏疾呻,在load函數(shù)調(diào)用registerDynamicModule:注冊(cè)改模塊除嘹。改宏可以穿入一個(gè)異步參數(shù),默認(rèn)為NO岸蜗,如果為YES則會(huì)在啟動(dòng)之后第一屏內(nèi)容展現(xiàn)之前異步執(zhí)行模塊的初始化尉咕,可以優(yōu)化啟動(dòng)時(shí)時(shí)間消耗。
(3)Annotation注冊(cè)
@BeeHiveMod(MainModule) //注冊(cè)MainModule
@interface MainModule ()<BHModuleProtocol>
@end
//注冊(cè)方法實(shí)現(xiàn)
#ifndef BeehiveModSectName
#define BeehiveModSectName "BeehiveMods"
#endif
#define BeeHiveDATA(sectname) __attribute((used, section("__DATA,"#sectname" ")))
#define BeeHiveMod(name) \
class BeeHive; char * k##name##_mod BeeHiveDATA(BeehiveMods) = ""#name"";
代碼段注冊(cè)璃岳,也就是直接操作代碼段年缎。感興趣的同學(xué)可以導(dǎo)出app編譯的源文件,可以在_data段看到BeehiveMods铃慷。具體怎么看app編譯后的可執(zhí)行文件单芜,請(qǐng)移步《iOS APP可執(zhí)行文件的組成》。
7.服務(wù)的注冊(cè)方式犁柜?
(1)靜態(tài)注冊(cè)
(2)API調(diào)用
- (void)modSetUp:(BHContext *)context {
[[BHServiceManager sharedManager] registerService: @protocol(PersonServiceProtocol) implClass: [PersonModuleService class]];
}
(3)Annotation注冊(cè)
@BeeHiveService(MainServiceProtocol, MainModuleService)
@interface MainModuleService ()<MainServiceProtocol>
@end
#ifndef BeehiveServiceSectName
#define BeehiveServiceSectName "BeehiveServices"
#endif
#define BeeHiveDATA(sectname) __attribute((used, section("__DATA,"#sectname" ")))
#define BeeHiveService(servicename,impl) \
class BeeHive;char * k##servicename##_service BeeHiveDATA(BeehiveServices) = "{ \""#servicename"\" : \""#impl"\"}";
3.BeeHive的項(xiàng)目使用
1.先引入BeeHive庫(kù)洲鸠,可以直接復(fù)制,也可以使用pod來(lái)管理馋缅。
2.把系統(tǒng)生成的Appdelegate干掉扒腕,新建一個(gè)繼承BHAppdelegate的子類(lèi)绢淀,把它改成入口類(lèi)。
3.配置BHContext袜匿,如下圖:
[BHContext shareInstance].application = application;
[BHContext shareInstance].launchOptions = launchOptions;
[BHContext shareInstance].env = BHEnvironmentDev;
[BHContext shareInstance].moduleConfigName = @"BeeHive.bundle/BeeHive-Jashion";
[BHContext shareInstance].serviceConfigName = @"BeeHive.bundle/BHService-Jashion";
moduleConfigName設(shè)置本地模塊加載地址更啄,serviceConfigName設(shè)置本地服務(wù)加載地址。
4.加載配置居灯,初始化BeeHive
[BeeHive shareInstance].enableException = YES;
[[BeeHive shareInstance] setContext: [BHContext shareInstance]];
5.設(shè)置rootController
id<MainServiceProtocol> mainVC = [[BeeHive shareInstance] createService: @protocol(MainServiceProtocol)];
if ([mainVC isKindOfClass: [UIViewController class]]) {
UINavigationController *navCtrl = [[UINavigationController alloc] initWithRootViewController: (UIViewController *)mainVC];
self.window = [[UIWindow alloc] initWithFrame: [UIScreen mainScreen].bounds];
self.window.rootViewController = navCtrl;
[self.window makeKeyAndVisible];
}
6.模塊之間的調(diào)用
id<MainServiceProtocol> mainVC = [[BeeHive shareInstance] createService: @protocol(MainServiceProtocol)];
通過(guò)BeeHive獲取相應(yīng)的實(shí)例祭务,然后通過(guò)protocol調(diào)用service暴露出來(lái)的方法。
總結(jié):##
我覺(jué)得整個(gè)架構(gòu)不是很清晰怪嫌,也有可能初步接觸义锥,還不太深入了解。解耦岩灭,我只看到了service之間的調(diào)用解耦拌倍,但是都依賴了協(xié)議。這個(gè)比運(yùn)用runtime的反射機(jī)制來(lái)解耦函數(shù)間的調(diào)用的好處在于噪径,可以在編譯期間發(fā)現(xiàn)錯(cuò)誤柱恤,而非運(yùn)行時(shí)。而且找爱,只要注冊(cè)了的module和service的實(shí)例就會(huì)一直被BeeHive持有梗顺,這可能會(huì)造成資源的浪費(fèi)。繼續(xù)觀望中车摄。