亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > iOS > 正文

iOS擼一個簡單路由Router的實現代碼

2019-10-21 18:39:48
字體:
來源:轉載
供稿:網友

平常開發中用戶點擊頭像, 進入個人主頁,這看似平常的操作, 背后極有可能會牽扯到多個模塊。 再如: 視頻模塊的播放頁, 有與視頻相關的音樂,點擊這些音樂,需要跳轉到音樂模塊的播放頁, 這樣視頻與音樂模塊之間,不可避免的會產生依賴或耦合。 這個問題讓人腦殼疼,相信很多朋友都這樣做過,寫一些代理或通知, 不停的傳遞事件; 有時干脆直接導入另一個模塊。

因為我在公司獨立開發, 顧及少一點,可以拿公司項目做實踐,在嘗試組件化的過程中, 了解到了路由, 對于解決上述問題, 有極大的幫助。因此我想總結并與大家分享一下。

什么是移動端路由層:

路由層的概念在服務端是指url請求的分層解析,將一個請求分發到對應的應用處理程序。移動端的路由層指的是將諸如App內頁面訪問、H5與App訪問的訪問請求和App間的訪問請求,進行分發處理的邏輯層。

移動端路由層需要解決的問題:

1.對外部提供遠程訪問的功能,實現跨應用調用響應,包括H5應用調用、其他App應用調用、系統訪問調用等
2.原生頁面、模塊、組件等定義,統稱為資源(Resource),在跨應用調用和路由層在不同端實現的業務表現需要一致的前提下,需要對資源進行定義,在路由提供內部請求分發的時候則可以提供不依賴對外進行資源定義的功能
3.外部調用如何使用統一標示(Uniform)進行表示資源
4.如何在移動端統一定義訪問請求的過程,從而達成移動端與web端的統一性
5.如何更好的兼容iOS、Android的系統訪問機制、App鏈接協議、web端路由機制與前端開發規范等
6.如何兼容各平臺(Android、iOS)App頁面導航機制
7.如何解決安全訪問問題
8.移動端在客戶端進行動態配置

移動端路由所應用的場景:

0.H5頁面與App原生頁面、模塊與組件的交互
1.App與App之間的相互訪問
2.App內部頁面跳轉、模塊調度與組件加載等
3.推送與通知系統解除硬編碼的邏輯,動態訪問原生資源,更好的支持通過通知和推送完成動態頁面訪問和邏輯執行
4.Extension等動態調用主App的資源
5.App實現更復雜的架構MVVM或者是VIPER架構,提供解除業務相互依賴的能力
6.以組件化為目的的工程改造,隔離各個業務,以制作單獨的組件

接口預覽

Router

NS_ASSUME_NONNULL_BEGIN@interface SJRouter : NSObject+ (instancetype)shared;- (void)handleRequest:(SJRouteRequest *)request completionHandler:(SJCompletionHandler)completionHandler;@endNS_ASSUME_NONNULL_END

RouteRequest

NS_ASSUME_NONNULL_BEGIN@interface SJRouteRequest : NSObject- (instancetype)initWithURL:(NSURL *)URL;- (instancetype)initWithPath:(NSString *)requestPath parameters:(nullable SJParameters)parameters;@property (nonatomic, strong, readonly) NSString *requestPath;@property (nonatomic, strong, readonly, nullable) SJParameters prts;- (instancetype)init NS_UNAVAILABLE;+ (instancetype)new NS_UNAVAILABLE;@endNS_ASSUME_NONNULL_END

RouteHandlerProtocol

NS_ASSUME_NONNULL_BEGINtypedef id SJParameters;@protocol SJRouteHandler+ (NSString *)routePath;+ (void)handleRequestWithParameters:(nullable SJParameters)parameters topViewController:(UIViewController *)topViewController completionHandler:(nullable SJCompletionHandler)completionHandler;@endNS_ASSUME_NONNULL_END

流程

簡單的講,app應用中,路由識別一個請求, 將它分派給對應的handler進行處理。 這個流程非常像發送一個網絡請求(拼接參數=>發起請求=>回調)。

同樣的,當Router收到下面的請求時(請求視頻播放頁):

- (void)push:(id)sender {  SJRouteRequest *request = [[SJRouteRequest alloc] initWithPath:@"video/playbackInfo" parameters:@{@"video_id":@(111)}];  [SJRouter.shared handleRequest:request completionHandler:^(id _Nullable result, NSError * _Nullable error) {#ifdef DEBUG    NSLog(@"%d - %s", (int)__LINE__, __func__);#endif  }];}

會嘗試識別路由, 找到匹配的handler,傳遞必要參數:

@implementation SJRouter- (void)handleRequest:(SJRouteRequest *)request completionHandler:(SJCompletionHandler)completionHandler {  NSParameterAssert(request); if ( !request ) return;  Class<SJRouteHandler> handler = _handlersM[request.requestPath];  if ( handler ) {    [handler handleRequestWithParameters:request.requestPath topViewController:_sj_get_top_view_controller() completionHandler:completionHandler];  }  else {    printf("/n (-_-) Unhandled request: %s", request.description.UTF8String);  }}@end

最后handler進行處理。

@implementation TestViewController+ (NSString *)routePath {  return @"video/playbackInfo";}+ (void)handleRequestWithParameters:(nullable SJParameters)parameters topViewController:(UIViewController *)topViewController completionHandler:(nullable SJCompletionHandler)completionHandler {  TestViewController *vc = [TestViewController new];  vc.completionHandler = completionHandler;  [topViewController.navigationController pushViewController:vc animated:YES];}@end

至此, 我們再回過頭看剛開始舉的那個例子:

視頻模塊的播放頁, 有與視頻相關的音樂,點擊這些音樂,需要跳轉到音樂模塊的播放頁。

此時,可以讓視頻模塊依賴Router, 進行跳轉請求。這看起來都是依賴,實則兩者差別很大了。

  1. 路由不止能處理跳轉音樂模塊的請求, 依賴也從多個變成只依賴Router即可。。。
  2. 在刪除某個依賴模塊時, 需要刪除依賴的代碼, 很煩的, 對吧。
  3. 吧啦吧啦吧啦吧啦吧啦。。。

所以點擊跳轉音樂模塊,可以替換成如下操作, 發起請求:

  SJRouteRequest *request = [[SJRouteRequest alloc] initWithPath:@"audio/playbackInfo" parameters:@{@"audio_id":@(232)}];  [SJRouter.shared handleRequest:request completionHandler:^(id _Nullable result, NSError * _Nullable error) {#ifdef DEBUG    NSLog(@"%d - %s", (int)__LINE__, __func__);#endif  }];

router找到對應的handler, 讓其進行處理。

Handler

從開始到現在,可以看出Handler就是最終執行請求的那個家伙。 相信大家都有疑問, 如何成為一個Handler?

很簡單,它是自動的(參見Router), 只要某個類遵守了SJRouteHandlerProtocol, 它便成為了一個Handler。再來看一遍協議吧。

NS_ASSUME_NONNULL_BEGINtypedef id SJParameters;@protocol SJRouteHandler+ (NSString *)routePath;+ (void)handleRequestWithParameters:(nullable SJParameters)parameters topViewController:(UIViewController *)topViewController completionHandler:(nullable SJCompletionHandler)completionHandler;@endNS_ASSUME_NONNULL_END
  1. routePath: 即路徑, 表示handler能夠處理的路徑。當發起請求時, Router會通過路徑獲取到對應的handler, 交給其進行處理。
  2. handleRequestWithParameters。。。: handler進行的處理。

Router

在整個請求過程中,Router做的事情實質上就是在眾多Handler中尋找命中注定的那一個。如何尋找呢?為什么遵守了SJRouteHandlerProtocol便自動成為了Handler呢? 

這自然要歸功于Runtime的強大力量,我們先看如何實現吧。

@implementation SJRouter - (instancetype)init {  self = [super init];  if ( !self ) return nil;  _handlersM = [NSMutableDictionary new];  int count = objc_getClassList(NULL, 0);  Class *classes = (Class *)malloc(sizeof(Class) * count); objc_getClassList(classes, count);  Protocol *p_handler = @protocol(SJRouteHandler);  for ( int i = 0 ; i < count ; ++ i ) {    Class cls = classes[i];    for ( Class thisCls = cls ; nil != thisCls ; thisCls = class_getSuperclass(thisCls) ) {      if ( !class_conformsToProtocol(thisCls, p_handler) ) continue;      if ( ![(id)thisCls respondsToSelector:@selector(routePath)] ) continue;      if ( ![(id)thisCls respondsToSelector:@selector(handleRequestWithParameters:topViewController:completionHandler:)] ) continue;      _handlersM[[(id<SJRouteHandler>)thisCls routePath]] = thisCls;      break;    }  }  if ( classes ) free(classes);  return self;}@end
  1. objc_getClassList: 很明顯了, 獲取App所有類。
  2. class_conformsToProtocol: 該類是否遵守某個協議。

得益于Runtime的這兩個函數,即可獲取到眾多的Handler。 當發起請求時, 在眾多Handler中尋找注定的那一個, 豈不是易如反掌。

Request

App發起的跳轉請求,更多到是內部頁面之間的跳轉, 我們最需要關注的就是它的路徑,所以整個路由都是圍繞著路徑去跳轉的, 而像URL中的scheme和host,體現出來的作用倒是不大。至少在我的項目中跳轉第三方App(例如分享)都是使用的友盟這類的SDK去處理的。

因此, Request持有每個請求的路徑, 以及必要的參數, 之后再無多余操作。 

好了, 就到這里了。

下面是項目地址, 有興趣到話可以與我一起交流哦。。。

https://github.com/changsanjiang/SJRouter

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VEVB武林網。


注:相關教程知識閱讀請移步到IOS開發頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产精品旅馆在线| 久久影视电视剧免费网站清宫辞电视| 中文字幕亚洲激情| 久久久在线免费观看| 成人免费高清完整版在线观看| 在线观看免费高清视频97| 国产丝袜高跟一区| 成人国产精品一区二区| 日韩欧美在线中文字幕| 97视频在线观看亚洲| 97人洗澡人人免费公开视频碰碰碰| 久久精品青青大伊人av| 国产精品成人免费电影| 国产精品亚洲欧美导航| 成年无码av片在线| 欧美日韩中文字幕综合视频| 欧美性xxxxxxxxx| 国产成人av在线播放| 成人午夜一级二级三级| 成人免费网站在线观看| 在线日韩欧美视频| 国内精品久久久久影院 日本资源| 久久久久久久久久av| 成人写真视频福利网| 精品视频在线播放色网色视频| 2019精品视频| 国产va免费精品高清在线观看| 成人久久精品视频| 色爱av美腿丝袜综合粉嫩av| 亚洲精品国产精品自产a区红杏吧| 国产精品久久久久久久久久尿| 国产精品视频26uuu| 亚洲国产精品大全| 国产精品 欧美在线| 亚洲国产成人一区| 国产91在线播放九色快色| 海角国产乱辈乱精品视频| 一区国产精品视频| 亚洲精品电影网在线观看| 国产福利视频一区| 日韩电影免费观看在线观看| 国产精品91在线| 久久精品视频一| 97精品视频在线观看| 国产精品久久久久久久久久久久久久| 国产精品国产三级国产专播精品人| 中文字幕亚洲欧美| 国产精品欧美在线| 亚洲激情免费观看| 色妞在线综合亚洲欧美| 欧美日韩中文字幕综合视频| wwwwwwww亚洲| 久久久女女女女999久久| 欧美极品少妇与黑人| 一区二区成人精品| 92国产精品久久久久首页| 成人精品一区二区三区电影免费| 日韩精品极品在线观看播放免费视频| 色七七影院综合| 国产精品视频一区二区三区四| 欧美精品18videos性欧| 欧美肥臀大乳一区二区免费视频| 欧美成aaa人片免费看| 日韩欧美中文在线| 亚洲乱码av中文一区二区| 精品一区二区电影| 青青草原一区二区| 91免费视频网站| 日韩欧美有码在线| 欧美激情亚洲一区| 国产999精品| 日韩av在线免播放器| 国产精品欧美亚洲777777| 精品无人区太爽高潮在线播放| 岛国av一区二区三区| 国产精品久久久久久久久久久不卡| 亚洲男人天堂九九视频| 一区二区三区视频免费在线观看| 国产精品免费一区豆花| 91视频国产高清| 精品国产视频在线| 亚洲男人的天堂在线播放| 久久夜色精品国产亚洲aⅴ| 欧美黑人一区二区三区| 久久久成人av| 亚洲自拍偷拍在线| 黄色成人在线播放| 国产精品视频久久久| 久久久久久久久久久免费精品| 色久欧美在线视频观看| 国产精品美女免费| 国产欧美一区二区三区视频| 日韩精品视频在线免费观看| 国产一区二区视频在线观看| 一区二区三区亚洲| 亚洲精品资源美女情侣酒店| 国产精品三级美女白浆呻吟| 亚洲最大福利网| 国产精品三级美女白浆呻吟| 国产欧美精品在线播放| 91九色国产在线| 中文字幕亚洲无线码a| 日韩黄色高清视频| 国产精品入口免费视| 久久免费在线观看| 亚洲国产成人av在线| 91经典在线视频| 国产综合香蕉五月婷在线| 日韩精品在线电影| 久久久久久久久国产精品| 国产一区二区av| 最新69国产成人精品视频免费| 92国产精品视频| 国产精品成久久久久三级| 日韩大片免费观看视频播放| 欧美电影在线播放| 日韩中文在线中文网三级| 国产精品人成电影| 中文字幕视频一区二区在线有码| 国产亚洲精品美女久久久久| 成人免费网视频| 好吊成人免视频| 久久99精品久久久久久青青91| 精品国产91久久久| 中文国产成人精品久久一| 北条麻妃一区二区在线观看| 日韩精品极品毛片系列视频| 亚洲人成伊人成综合网久久久| 国产精品一区二区久久| 久久综合网hezyo| 久久精品99久久久久久久久| 亚洲高清免费观看高清完整版| 亚洲精品动漫100p| 欧美午夜无遮挡| 亚洲国产美女精品久久久久∴| 亚洲v日韩v综合v精品v| 亚洲九九九在线观看| 国产精品国语对白| 欧美成人一区在线| 亚洲成人在线视频播放| 91精品免费久久久久久久久| 国产精品高潮呻吟久久av无限| 亚洲福利视频网| 久久久久国色av免费观看性色| 少妇av一区二区三区| 欧美日韩中文在线观看| 2019中文字幕在线免费观看| 黑人巨大精品欧美一区免费视频| 国外成人在线视频| 国内精久久久久久久久久人| 国产精品自产拍在线观看中文| 日韩性生活视频| 成人黄色午夜影院| 精品露脸国产偷人在视频| 日韩hd视频在线观看| 亚洲自拍在线观看| 中文字幕亚洲第一| 国产欧美一区二区三区四区| 一区三区二区视频| 国产免费久久av| 一区二区三区天堂av| 国产精品亚洲欧美导航| 久热精品在线视频| 久久天堂av综合合色|