怎么在iOS中實(shí)現(xiàn)路由Router

怎么在iOS中實(shí)現(xiàn)路由Router?很多新手對(duì)此不是很清楚,為了幫助大家解決這個(gè)難題,下面小編將為大家詳細(xì)講解,有這方面需求的人可以來(lái)學(xué)習(xí)下,希望你能有所收獲。

創(chuàng)新互聯(lián)建站專(zhuān)注于呼瑪企業(yè)網(wǎng)站建設(shè),成都響應(yīng)式網(wǎng)站建設(shè)公司,商城系統(tǒng)網(wǎng)站開(kāi)發(fā)。呼瑪網(wǎng)站建設(shè)公司,為呼瑪?shù)鹊貐^(qū)提供建站服務(wù)。全流程按需定制制作,專(zhuān)業(yè)設(shè)計(jì),全程項(xiàng)目跟蹤,創(chuàng)新互聯(lián)建站專(zhuān)業(yè)和態(tài)度為您提供的服務(wù)

什么是移動(dòng)端路由層:

路由層的概念在服務(wù)端是指url請(qǐng)求的分層解析,將一個(gè)請(qǐng)求分發(fā)到對(duì)應(yīng)的應(yīng)用處理程序。移動(dòng)端的路由層指的是將諸如App內(nèi)頁(yè)面訪問(wèn)、H5與App訪問(wèn)的訪問(wèn)請(qǐng)求和App間的訪問(wèn)請(qǐng)求,進(jìn)行分發(fā)處理的邏輯層。

移動(dòng)端路由層需要解決的問(wèn)題:

1.對(duì)外部提供遠(yuǎn)程訪問(wèn)的功能,實(shí)現(xiàn)跨應(yīng)用調(diào)用響應(yīng),包括H5應(yīng)用調(diào)用、其他App應(yīng)用調(diào)用、系統(tǒng)訪問(wèn)調(diào)用等
2.原生頁(yè)面、模塊、組件等定義,統(tǒng)稱(chēng)為資源(Resource),在跨應(yīng)用調(diào)用和路由層在不同端實(shí)現(xiàn)的業(yè)務(wù)表現(xiàn)需要一致的前提下,需要對(duì)資源進(jìn)行定義,在路由提供內(nèi)部請(qǐng)求分發(fā)的時(shí)候則可以提供不依賴(lài)對(duì)外進(jìn)行資源定義的功能
3.外部調(diào)用如何使用統(tǒng)一標(biāo)示(Uniform)進(jìn)行表示資源
4.如何在移動(dòng)端統(tǒng)一定義訪問(wèn)請(qǐng)求的過(guò)程,從而達(dá)成移動(dòng)端與web端的統(tǒng)一性
5.如何更好的兼容iOS、Android的系統(tǒng)訪問(wèn)機(jī)制、App鏈接協(xié)議、web端路由機(jī)制與前端開(kāi)發(fā)規(guī)范等
6.如何兼容各平臺(tái)(Android、iOS)App頁(yè)面導(dǎo)航機(jī)制
7.如何解決安全訪問(wèn)問(wèn)題
8.移動(dòng)端在客戶端進(jìn)行動(dòng)態(tài)配置

移動(dòng)端路由所應(yīng)用的場(chǎng)景:

0.H5頁(yè)面與App原生頁(yè)面、模塊與組件的交互
1.App與App之間的相互訪問(wèn)
2.App內(nèi)部頁(yè)面跳轉(zhuǎn)、模塊調(diào)度與組件加載等
3.推送與通知系統(tǒng)解除硬編碼的邏輯,動(dòng)態(tài)訪問(wèn)原生資源,更好的支持通過(guò)通知和推送完成動(dòng)態(tài)頁(yè)面訪問(wèn)和邏輯執(zhí)行
4.Extension等動(dòng)態(tài)調(diào)用主App的資源
5.App實(shí)現(xiàn)更復(fù)雜的架構(gòu)MVVM或者是VIPER架構(gòu),提供解除業(yè)務(wù)相互依賴(lài)的能力
6.以組件化為目的的工程改造,隔離各個(gè)業(yè)務(wù),以制作單獨(dú)的組件

接口預(yù)覽

Router

NS_ASSUME_NONNULL_BEGIN
@interface SJRouter : NSObject
+ (instancetype)shared;

- (void)handleRequest:(SJRouteRequest *)request completionHandler:(SJCompletionHandler)completionHandler;
@end
NS_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;
@end
NS_ASSUME_NONNULL_END

RouteHandlerProtocol

NS_ASSUME_NONNULL_BEGIN
typedef id SJParameters;

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

流程

簡(jiǎn)單的講,app應(yīng)用中,路由識(shí)別一個(gè)請(qǐng)求, 將它分派給對(duì)應(yīng)的handler進(jìn)行處理。 這個(gè)流程非常像發(fā)送一個(gè)網(wǎng)絡(luò)請(qǐng)求(拼接參數(shù)=>發(fā)起請(qǐng)求=>回調(diào))。

同樣的,當(dāng)Router收到下面的請(qǐng)求時(shí)(請(qǐng)求視頻播放頁(yè)):

- (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
  }];
}

會(huì)嘗試識(shí)別路由, 找到匹配的handler,傳遞必要參數(shù):

@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進(jìn)行處理。

@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

至此, 我們?cè)倩剡^(guò)頭看剛開(kāi)始舉的那個(gè)例子:

視頻模塊的播放頁(yè), 有與視頻相關(guān)的音樂(lè),點(diǎn)擊這些音樂(lè),需要跳轉(zhuǎn)到音樂(lè)模塊的播放頁(yè)。

此時(shí),可以讓視頻模塊依賴(lài)Router, 進(jìn)行跳轉(zhuǎn)請(qǐng)求。這看起來(lái)都是依賴(lài),實(shí)則兩者差別很大了。

  1. 路由不止能處理跳轉(zhuǎn)音樂(lè)模塊的請(qǐng)求, 依賴(lài)也從多個(gè)變成只依賴(lài)Router即可。。。

  2. 在刪除某個(gè)依賴(lài)模塊時(shí), 需要?jiǎng)h除依賴(lài)的代碼, 很煩的, 對(duì)吧。

  3. 吧啦吧啦吧啦吧啦吧啦。。。

所以點(diǎn)擊跳轉(zhuǎn)音樂(lè)模塊,可以替換成如下操作, 發(fā)起請(qǐng)求:

  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找到對(duì)應(yīng)的handler, 讓其進(jìn)行處理。

Handler

從開(kāi)始到現(xiàn)在,可以看出Handler就是最終執(zhí)行請(qǐng)求的那個(gè)家伙。 相信大家都有疑問(wèn), 如何成為一個(gè)Handler?

很簡(jiǎn)單,它是自動(dòng)的(參見(jiàn)Router), 只要某個(gè)類(lèi)遵守了SJRouteHandlerProtocol, 它便成為了一個(gè)Handler。再來(lái)看一遍協(xié)議吧。

NS_ASSUME_NONNULL_BEGIN
typedef id SJParameters;

@protocol SJRouteHandler
+ (NSString *)routePath;
+ (void)handleRequestWithParameters:(nullable SJParameters)parameters topViewController:(UIViewController *)topViewController completionHandler:(nullable SJCompletionHandler)completionHandler;
@end
NS_ASSUME_NONNULL_END
  1. routePath: 即路徑, 表示handler能夠處理的路徑。當(dāng)發(fā)起請(qǐng)求時(shí), Router會(huì)通過(guò)路徑獲取到對(duì)應(yīng)的handler, 交給其進(jìn)行處理。

  2. handleRequestWithParameters。。。: handler進(jìn)行的處理。

Router

在整個(gè)請(qǐng)求過(guò)程中,Router做的事情實(shí)質(zhì)上就是在眾多Handler中尋找命中注定的那一個(gè)。如何尋找呢?為什么遵守了SJRouteHandlerProtocol便自動(dòng)成為了Handler呢?

這自然要?dú)w功于Runtime的強(qiáng)大力量,我們先看如何實(shí)現(xiàn)吧。

@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所有類(lèi)。

  2. class_conformsToProtocol: 該類(lèi)是否遵守某個(gè)協(xié)議。

看完上述內(nèi)容是否對(duì)您有幫助呢?如果還想對(duì)相關(guān)知識(shí)有進(jìn)一步的了解或閱讀更多相關(guān)文章,請(qǐng)關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝您對(duì)創(chuàng)新互聯(lián)的支持。

本文題目:怎么在iOS中實(shí)現(xiàn)路由Router
標(biāo)題網(wǎng)址:http://muchs.cn/article44/ipghhe.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供標(biāo)簽優(yōu)化、網(wǎng)站排名、企業(yè)網(wǎng)站制作、App開(kāi)發(fā)、網(wǎng)站導(dǎo)航、微信小程序

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)

綿陽(yáng)服務(wù)器托管