使用組件化是爲了解耦處理,多個模塊之間經過協議進行交互。而負責解析協議,找到目的控制器,或者是返回對象給調用者的這個組件就是路由組件。本文講解如何使用核心的50行代碼實現一個路由組件。html
以前看過挺多的關於路由管理、路由處理的文章,經常會和組件化出如今一塊兒,一開始不知道爲什麼路由和組件化出如今一塊兒,後來公司的項目中使用了路由組件(他自己也是一個組件,確切的說是一箇中間人或者中介者),才忽然想明白了,原來如此。
使用組件化是爲了解耦處理,多個模塊之間經過協議進行交互。而負責解析協議,找到目的控制器,或者是返回對象給調用者的這個組件就是路由組件。ios
路由組件的職責主要是:git
下面會會在以上分析的基礎上實現一個簡單的路由組件,對應的代碼能夠在YTRouterDemo這裏找到數據庫
路由的實現包括兩部分:路由註冊實現以及路由使用實現數組
路由註冊實現時序圖:
 緩存
如上圖所示,步驟很簡單:架構
YTRouterActionObject
對象,用於保存path和對應的blokYTRouterActionObject
對象保存在上一步找到的位置中以上步驟對應的代碼以下:組件化
- (void)registerPath:(NSString *)path actionBlock:(RouterActionBlock)actionBlock {
YTRouterActionObject *actionObject = [YTRouterActionObject new];
actionObject.path = path;
actionObject.actionBlock = actionBlock;
NSMutableDictionary *subRouter = [self subRouterMapWithPath:path];
subRouter[YTRouterActionObjectKey] = actionObject;
}
- (NSMutableDictionary *)subRouterMapWithPath:(NSString *)path {
NSArray *components = [path componentsSeparatedByString:@"/"];
NSMutableDictionary *subRouter = self.routerMap;
for (NSString *component in components) {
if (component.length == 0) {
continue;
}
if (!subRouter[component]) {
subRouter[component] = [NSMutableDictionary new];
}
subRouter = subRouter[component];
}
return subRouter;
}
複製代碼
在Demo中註冊的幾個路由最終的配置以下,好比home/messagelist
對應的路由配置保存在<YTRouterActionObject: 0x6040000365e0>
對象中性能
Printing description of self->_routerMap:
{
home = {
"_" = "<YTRouterActionObject: 0x60c00003b040>";
messagelist = {
"_" = "<YTRouterActionObject: 0x6040000365e0>";
detail = {
"_" = "<YTRouterActionObject: 0x600000038ec0>";
};
getmessage = {
"_" = "<YTRouterActionObject: 0x600000038e80>";
};
};
};
}
複製代碼
路由使用實現時序圖:
 測試
如上圖所示,步驟很簡單:
YTRouterActionObject
對象YTRouterActionObject
對象的actionBlock,會傳遞一個YTRouterActionCallbackObject
對象,若是調用者須要的是返回值,可使用YTRouterActionCallbackObject
對象的actionCallbackBlock
傳遞一個返回值,這個actionBlock是又業務方的註冊者實現的以上步驟對應的代碼以下:
- (BOOL)runWithActionCallbackObject:(YTRouterActionCallbackObject *)actionCallbackObject {
// 判斷是否支持scheme
if (![self canAcceptScheme:actionCallbackObject.uri.scheme]) {
return NO;
}
// 獲取path對應的ActionObject
YTRouterActionObject *actionObject = [self actionObjectWithPath:actionCallbackObject.uri.path];
// 執行Path註冊的對應Block
!actionObject.actionBlock ?: actionObject.actionBlock(actionCallbackObject);
return YES;
}
- (YTRouterActionObject *)actionObjectWithPath:(NSString *)path {
NSMutableDictionary *subRouter = [self subRouterMapWithPath:path];
return subRouter[YTRouterActionObjectKey];
}
複製代碼
以上講到了核心的路由註冊實現
和路由使用實現
,總共代碼尚未50行,因此仍是很簡單的,接下來會講下客戶端的使用步驟,包括
註冊的時機須要比較找,考慮到集成的方便,選擇在load方法中處理路由註冊,以下代碼所示,添加了幾個測試的路由,分兩種狀況來講明下使用
一、不須要返回值
以下注冊"home/messagelist"
的是一個頁面跳轉的路由,actionBlock的參數是一個YTRouterActionCallbackObject
對象,能夠從YTRouterActionCallbackObject
對象或者到參數,關於如何傳遞值,會在下面的客戶端調用者使用
這裏講到。而後在actionBlock處理目的頁面的初始化、參數設置等步驟,而後執行頁面跳轉。
二、須要返回值
以下注冊"home/messagelist/getmessage"
的是一個提供返回值的路由,一樣也能夠從YTRouterActionCallbackObject
對象獲取參數,另外YTRouterActionCallbackObject
對象還有一個actionCallbackBlock
屬性是專門處理返回參數給調用者的,以下的代碼只是簡單返回一個字符串,在更加具體的業務場景中,這裏會設置接口調用、數據庫查詢等任務,最後把結果返回。
@implementation ModuleAUriRegister
+ (void)load {
[[YTRouterManager sharedRouterManager] registerPath:@"home/messagelist" actionBlock:^(YTRouterActionCallbackObject *callbackObject) {
MessageListViewController *messageListVC = [MessageListViewController new];
NSString *title = callbackObject.uri.params[@"title"];
messageListVC.title = title;
[[UIViewController yt_currentViewControlloer].navigationController pushViewController:messageListVC animated:YES];;
}];
[[YTRouterManager sharedRouterManager] registerPath:@"home/" actionBlock:^(YTRouterActionCallbackObject *callbackObject) {
}];
[[YTRouterManager sharedRouterManager] registerPath:@"home/messagelist/detail" actionBlock:^(YTRouterActionCallbackObject *callbackObject) {
}];
[[YTRouterManager sharedRouterManager] registerPath:@"home/messagelist/getmessage" actionBlock:^(YTRouterActionCallbackObject *callbackObject) {
// 內容回調
!callbackObject.actionCallbackBlock ?: callbackObject.actionCallbackBlock(@"message content text demo");
}];
}
@end
複製代碼
一、簡單的path跳轉調用
使用YTRouterManager
單例對象的runWithPath
方法,傳遞一個註冊的path參數完成跳轉。
[self addActionWithTitle:@"Router頁面跳轉" detailText:@"home/messagelist" callback:^{
[[YTRouterManager sharedRouterManager] runWithPath:@"home/messagelist"];
}];
複製代碼
二、使用URL調用和有URL參數的調用
使用YTRouterManager
單例對象的runWithURLString
方法,傳遞一個完整的包含了scheme/path,或者有參數的會纔有參數的URL,好比"YTRouter://home/messagelist"
和 "YTRouter://home/messagelist?title=Hello Message"
,路由組件會解析出裏面的scheme、path、params,進行scheme過濾處理、path查詢YTRouterActionObject
對象處理、參數傳遞處理。
[self addActionWithTitle:@"Router使用URL調用" detailText:@"YTRouter://home/messagelist" callback:^{
[[YTRouterManager sharedRouterManager] runWithURLString:@"YTRouter://home/messagelist"];
}];
[self addActionWithTitle:@"Router使用帶參數的URL調用" detailText:@"YTRouter://home/messagelist?title=Hello Message" callback:^{
[[YTRouterManager sharedRouterManager] runWithURLString:@"YTRouter://home/messagelist?title=Hello Message"];
}];
複製代碼
效果以下圖所示:
三、簡單的path跳轉調用
使用YTRouterManager
單例對象的runWithActionCallbackObject
方法,傳遞一個YTRouterActionCallbackObject
類型的參數,設置YTRouterActionCallbackObject
對象的uri和結果回調actionCallbackBlock參數,在actionCallbackBlock中處理返回值。
[self addActionWithTitle:@"Router獲取返回值" detailText:@"home/messagelist/getmessage" callback:^{
__block id message = nil;
YTRouterActionCallbackObject *actionCallbackObject = [YTRouterActionCallbackObject new];
actionCallbackObject.uri = [[YTUri alloc] initWithPath:@"home/messagelist/getmessage"];
actionCallbackObject.actionCallbackBlock = ^(id result) {
message = result;
};
[[YTRouterManager sharedRouterManager] runWithActionCallbackObject:actionCallbackObject];
NSLog(@"message = %@", message);
}];
複製代碼
iOS應用架構談 組件化方案
iOS組件化實踐方案-LDBusMediator煉就
iOS組件化思路-大神博客研讀和思考
iOS 組件化方案探索
蘑菇街 App 的組件化之路
NSRecursiveLock遞歸鎖的使用