当前位置: 移动技术网 > IT编程>移动开发>IOS > iOS撸一个简单路由Router的实现代码

iOS撸一个简单路由Router的实现代码

2019年07月23日  | 移动技术网IT编程  | 我要评论

vovose,aiqdy,台湾海军

平常开发中用户点击头像, 进入个人主页,这看似平常的操作, 背后极有可能会牵扯到多个模块。 再如: 视频模块的播放页, 有与视频相关的音乐,点击这些音乐,需要跳转到音乐模块的播放页, 这样视频与音乐模块之间,不可避免的会产生依赖或耦合。 这个问题让人脑壳疼,相信很多朋友都这样做过,写一些代理或通知, 不停的传递事件; 有时干脆直接导入另一个模块。

因为我在公司独立开发, 顾及少一点,可以拿公司项目做实践,在尝试组件化的过程中, 了解到了路由, 对于解决上述问题, 有极大的帮助。因此我想总结并与大家分享一下。

什么是移动端路由层:

路由层的概念在服务端是指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;
@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

流程

简单的讲,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_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能够处理的路径。当发起请求时, 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

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持移动技术网。

如对本文有疑问,请在下面进行留言讨论,广大热心网友会与你互动!! 点击进行留言回复

相关文章:

  • ios uicollectionview实现横向滚动

    现在使用卡片效果的app很多,之前公司让实现一种卡片效果,就写了一篇关于实现卡片的文章。文章最后附有demo实现上我选择了使用uicollectionview ... [阅读全文]
  • iOS UICollectionView实现横向滑动

    本文实例为大家分享了ios uicollectionview实现横向滑动的具体代码,供大家参考,具体内容如下uicollectionview的横向滚动,目前我使... [阅读全文]
  • iOS13适配深色模式(Dark Mode)的实现

    iOS13适配深色模式(Dark Mode)的实现

    好像大概也许是一年前, mac os系统发布了深色模式外观, 看着挺刺激, 时至今日用着也还挺爽的终于, 随着iphone11等新手机的发售, ios 13系统... [阅读全文]
  • ios 使用xcode11 新建项目工程的步骤详解

    ios 使用xcode11 新建项目工程的步骤详解

    xcode11新建项目工程,新增了scenedelegate这个类,转而将原appdelegate负责的对ui生命周期的处理担子接了过来。故此可以理解为:ios... [阅读全文]
  • iOS实现转盘效果

    本文实例为大家分享了ios实现转盘效果的具体代码,供大家参考,具体内容如下demo下载地址: ios转盘效果功能:实现了常用的ios转盘效果,轮盘抽奖效果的实现... [阅读全文]
  • iOS开发实现转盘功能

    本文实例为大家分享了ios实现转盘功能的具体代码,供大家参考,具体内容如下今天给同学们讲解一下一个转盘选号的功能,直接上代码直接看viewcontroller#... [阅读全文]
  • iOS实现轮盘动态效果

    本文实例为大家分享了ios实现轮盘动态效果的具体代码,供大家参考,具体内容如下一个常用的绘图,主要用来打分之类的动画,效果如下。主要是ios的绘图和动画,本来想... [阅读全文]
  • iOS实现九宫格连线手势解锁

    本文实例为大家分享了ios实现九宫格连线手势解锁的具体代码,供大家参考,具体内容如下demo下载地址:效果图:核心代码://// clockview.m// 手... [阅读全文]
  • iOS实现卡片堆叠效果

    本文实例为大家分享了ios实现卡片堆叠效果的具体代码,供大家参考,具体内容如下如图,这就是最终效果。去年安卓5.0发布的时候,当我看到安卓全新的material... [阅读全文]
  • iOS利用余弦函数实现卡片浏览工具

    iOS利用余弦函数实现卡片浏览工具

    本文实例为大家分享了ios利用余弦函数实现卡片浏览工具的具体代码,供大家参考,具体内容如下一、实现效果通过拖拽屏幕实现卡片移动,左右两侧的卡片随着拖动变小,中间... [阅读全文]
验证码:
移动技术网