TheRouter 是一个用于移动端 APP,包括 Android、iOS、Harmony 三端的模块化、组件化开发的一整套解决方案框架。提供了三端高一致性,对移动端开发者更友好,让开发人员更适应,使用起来也更顺手。在鸿蒙上, TheRouter 基于 HMRouter 做了深度定制,不仅支持平台化应用实现组件化、跨模块调用、动态化等功能的集成等功能基础上,还提供了编译时安全检查、支持动态路由下发与修改、路由 Path 一对多等高度动态能力。
Github: https://github.com/HuolalaTech/hll-wp-therouter-harmony/
官网:http://therouter.cn/
TheRouter Harmony 核心功能具备如下能力:
页面导航跳转能力 (Navigator)
跨模块依赖注入能力 (ServiceProvider)
动态化能力 (ActionManager)
<br>
一、为什么要使用 TheRouter
路由是现如今移动端开发中必不可少的功能,尤其是企业级 APP,可以用于将多模块页面跳转的强依赖关系解耦,同时减少跨团队开发的互相依赖问题。
对于大型 APP 开发,基本都会选用模块化 (或组件化) 方式开发,对于模块间解耦要求更高。 TheRouter 是一整套完全面向模块化开发的解决方案,不仅能支持常规的模块依赖解耦、页面跳转,同时提供了模块化过程中常见问题的解决办法。
<br>
1.1 TheRouter 鸿蒙端的三大能力
Navigator:
支持 Path 与页面多对一关系或一对一关系,可用于解决多端 path 统一问题
页面 Path 支持正则表达式声明
支持 json 格式路由表导出
路由表支持为页面添加注释说明
支持动态下发 json 路由表,降级任意页面为 H5
支持页面跳转拦截处理
支持使用路由跳转到第三方 SDK 中的页面
ServiceProvider:
支持跨模块依赖注入
支持自定义注入项的创建规则,依赖注入可自定义参数
支持注入对象缓存,多次注入,只会 new 一次对象
ActionManager:
支持全局回调配置
支持多对一链式响应
支持优先级响应
方法支持返回值与入参
支持记录调用路径,解决调试期观察者模式无法追踪 Observable 的问题
<br>
二、鸿蒙路由方案
根据华为官方文档建议,从 API 10 开始,推荐使用 NavPathStack 配合 navDestination 属性进行页面路由。所以 TheRouter 也是按照这个方案实现的,如果你的项目还是使用 ohos.router 组件,建议尽早迁移。
详细内容请查看华为官方文档:https://developer.huawei.com/consumer/cn/doc/harmonyos-references/js-apis-router
无论哪个平台,路由的本质就是一个 Map,或者说是字典。其中 key 是,页面的 path,value 是路由项。对于 Action、ServiceProvider,也都是一样。
所以在鸿蒙上最核心的重点,是实现在编译期将注解 path 关联到具体的路由项,使其产生一个一对多或一对一的对应关系。其次就是要考虑如何与系统自带的路由表兼容,遵循系统方案而不是完全打造一套,否则可能引起将来的不确定性。
在 TheRouter 中,通过编译期的 hvigor 插件,解析全部的注解关键字,并将获取到的内容保存下来,在应用编译完成后,参照系统的路由表格式,生成一份增量的路由表,聚合到系统的路由表内。正是因为这样,才能做到兼容第三方 SDK,正常跳转到第三方页面。既然第三方都能跳转,那么其他的二方、或自己的独立模块自然也就可以正常跳转了。
<br>
三、使用 TheRouter 页面跳转
3.1 声明路由项
如果一个页面允许被路由打开,则需要使用注解 @Route 声明路由项,每个页面允许声明多个路由项,也就是一对多的能力,极大降低多端路由统一时的业务影响面。
参数释义
path: 路由 path 【必传】。
建议是一个 url,并推荐三端 url 统一。path 内支持使用正则表达式,允许多个 path 对应同一个 Page。
description: 页面描述【可选】。
会被记录到路由表中,方便后期排查的时候知道每个 path 或 Page 是什么业务。
params: 页面参数【可选】。
自动写入当前页面参数中,允许写在路由表中动态下发修改默认值,或通过路由跳转时代码传入。
launchMode: 当前页的启动方式【可选】。
启动模式,可选项:'STANDARD'(默认)、'MOVE_TO_TOP_SINGLETON'、'POP_TO_SINGLETON'、'NEW_INSTANCE'。
@Route({ path: BaseConstant.MAIN_PAGE, description: 'Demo首页', params: ["hello", "路由表默认参数"], launchMode:'STANDARD' })
@Component
export struct MainPage {
}
3.2 发起页面跳转
传入的参数可以是 string 和基本数据类型、也可以是 ESObject 对象。
TheRouter.build("http://therouter.com/home")
.withNumber("key1", 12345678)
.withString("key2", "参数")
.withBoolean("key3", false)
.with({xxx:xxx})
// navigation、replace、pop 均可以额外传入 callback 参数,对当前跳转的个状态回调
.navigation();
// 替换页面(相当于先 pop 再 push)
.replace();
// 关闭当前页
.pop();
3.3 路由表生成规则
如果两条路由的 path 完全相同,则认为是同一条路由,不会考虑参数是否相同。
路由表生成规则:编译期按照如下顺序取并集。
覆盖规则:
根据如下顺序,如果相同,后者可以覆盖前者的路由表规则。
编译期解析注解生成路由表
首先取 业务模块(har/hap) 中的路由表
再取 主 hsp module 代码中的路由表
最后取 resources/base/profile/RouteMap.json 文件中声明的路由表。
如果编译期没有这个文件,会生成一份默认路由表放在这个目录内(编译完成后如果没有配置保留,会自动删掉);如果有,会将路由表合并。
运行时线上动态下发的路由表
路由表允许线上动态下发,将覆盖本地路由表,详见 【3.4 动态路由表的设计与使用】
如果编译期没有这个文件,会生成一份默认路由表放在这个目录内(编译完成后如果没有配置保留,会自动删掉);如果有,会将路由表合并,因此,对于没办法修改代码的第三方 SDK 内部,如果希望通过路由打开,只需要手动在 RouteMap.json 文件中声明,就能通过路由打开了。
<br>
3.4 动态路由表的设计与使用
TheRouter 的路由表是动态添加的,项目每次编译后,会在 app 内生成一份当前模块的全量路由表。这个路由表也可以后续通过远程下发的方式使用,例如远端可以针对不同的 APP 版本,下发不同的路由表达到配置目的。这样如果将来线上某些页面发生 Crash,可以通过将这个页面的落地页替换为 H5 的方式,临时解决这类问题。
<br>
有两种推荐的远程下发方式可供使用方选择:
将打包系统与配置系统打通,每次新版本打包后自动将所有模块 (hsp、har、hap) resource/rawfile/ 目录中的路由表文件上传到配置系统,聚合成一个 json 后,下发给对应版本 APP 。优点在于全自动不会出错。
配置系统无法打通,线上手动下发需要修改的路由项,因为 TheRouter 会自动用最新下发的路由项覆盖包内的路由项。优点在于精确,且流量资源占用小。
<br>
动态路由限制 :准确的说,应该是鸿蒙系统的限制。在鸿蒙上,路由表必须是静态的并且在编译期确定下来。TheRouter Harmony 做了一些黑科技处理,允许动态加载一个或多个路由表,但是动态加载的路由页面必须是在编译期就已经存在的,不能凭空新增(类似 Android 的 Activity,在编译后就不能再改或新增注册清单文件了)。
// 与Android逻辑不同,此代码 必须 在页面打开之前,路由初始化之后调用。 建议紧跟 TheRouter.init() 调用
TheRouter.setRouteMapInitTask(task: (map: Map<string, routeitem>) => void);
/**
* 此处的 map 就是当前应用的路由表全量,
* 当获取到远端路由表以后,把路由表继续传入map中,有重复项可自动覆盖
*/
TheRouter.setRouteMapInitTask(() => {
// 此处为纯业务逻辑,每家公司远端配置方案可能都不一样
const json = Connfig.doHttp("routeMap");
// 只需要将路由json返回给框架即可,不建议在任务中做耗时操作
return json;
});
3.5 拦截器用法
框架内置四种自定义处理器可供业务场景定制,用于在路由跳转过程中,以切面的方式统一修改路由落地页参数信息。
Harmony 路由与 Android 路由的拦截器使用完全一致,可以直接参考【Android 文档 第三部分】。
// 所有拦截器方法均在 TheRouter 类下,可以直接如下方式全局调用
TheRouter.addNavigatorPathFixHandle()
/**
* 应用场景:用于修复客户端上路由 path 错误问题。
* 例如:相对路径转绝对路径,或由于服务端下发的链接无法固定https或http,但客户端代码写死了 https 的 path,就可以用这种方式统一。
* 注:必须在 TheRouter.build() 方法调用前添加处理器,否则处理器前的所有path不会被修改。
*/
static addNavigatorPathFixHandle(handle: NavigatorPathFixHandle);
/**
* 页面替换器
* 应用场景:需要将某些path指定为新链接的时候使用。 也可以用在修复链接的场景,但是与 path 修改器不同的是,修改器通常是为了解决通用性的问题,替换器只在页面跳转时才会生效,更多是用来解决特性问题。
*
* 例如模块化的时候,首页壳模板组件中开发了一个SplashActivity广告组件作为应用的MainActivity,在闪屏广告结束的时候自动跳转业务首页页面。 但是每个业务不同,首页页面的 Path 也不相同,而不希望让每个业务线自己去改这个首页壳模板组件,此时就可以组件中先写占位符https://kymjs.com/splash/to/home,让接入方通过 Path 替换器解决。
* 注:必须在 TheRouter.build().navigation() 方法调用前添加处理器,否则处理器前的所有跳转不会被替换。
*/
static addPathReplaceInterceptor(interceptor: PathReplaceInterceptor);
/**
* 路由替换器
* 应用场景:常用在未登录不能使用的页面上。例如访问用户钱包页面,在钱包页声明的时候,可以在路由表上声明本页面是需要登录的,在路由跳转过程中,如果落地页是需要登录的,则先替换路由到登录页,同时将原落地页信息作为参数传给登录页,登录流程处理完成后可以继续执行之前的路由操作。
*
* 路由替换器的拦截点更靠后,主要用于框架已经从路由表中根据 path 找到路由以后,对找到的路由做操作。
*
* 这种逻辑在所有页面跳转前写不太合适,以前的做法通常是在落地页写逻辑判断用户是否具有权限,但其实在路由层完成更合适。
* 注:必须在 TheRouter.build().navigation() 方法调用前添加处理器,否则处理器前的所有跳转不会被替换。
*/
static addRouterReplaceInterceptor(interceptor: RouterReplaceInterceptor);
/**
* 路由AOP拦截器
* 与前三个处理器不同的点在于,路由的AOP拦截器全局只能有一个。用于实现AOP的能力,在整个TheRouter跳转的过程中,跳转前,目标页是否找到的回调,跳转时,跳转后,都可以做一些自定义的逻辑处理。
*
* 使用场景:场景很多,最常用的是可以拦截一些跳转,例如debug页面在生产环境不打开,或定制startActivity跳转方法。
*/
public setRouterInterceptor(interceptor: (route: RouteItem, callback: (route: RouteItem) => void) => void);
3.6 高级用法
TheRouter 同时支持更多页面跳转能力:
为第三方库里面的页面添加路由表,达到对某些页面降级替换的目的;
跳转过程拦截器(总共四层,可根据实际需求使用);
跳转结果回调;
<br>
四、跨模块依赖注入 ServiceProvider 的设计
对于模块化开发中跨模块的调用,我们推荐采用 SOA (面向服务架构) 的设计方式,服务调用方与使用方完全隔离,调用模块外的能力不需要关注能力的提供者是谁。
ServiceProvider 的核心设计思想也是这样的,目前服务间的调用协议采用接口的方式。当然,也可以兼容不通过接口下沉而是直接调用的情况。
<img src="https://oscimg.oschina.net/oscnet//2b415a4c5e28eb030692174056fbf876.jpeg" class="blog-img">
服务提供方负责提供服务,不需要关心调用方是谁会在何时调用自己。
服务的使用方只关注服务本身,不需要关心这个服务是谁提供的,只需要只能服务能提供哪些能力即可。
例如上面的图片:拉拉需要使用录音的服务,小货则向外提供一个录音的服务,由 TheRouter 的 ServiceProvider 负责撮合。
<br>
4.1 服务使用方:拉拉
她无需关心,IRecordService 这个接口服务是谁提供的,他只需要知道自己需要使用这样的一个服务就行了。
注:如果没有提供服务的提供方,TheRouter.get() 可能返回 undefined
TheRouter.get<irecordservice>(BaseConstant.CLASS_SERVICE)?.doRecord()
4.2 服务提供方:小货
服务提供方需要声明一个提供服务的方法,用 @ServiceProvider 注解标记,并需要实现接口 IServiceProvider。
// 类名不限定,任意名字都行
// 所有的 ServiceProvider 必须实现 IServiceProvider 接口
// 多次添加重复serviceName,框架会保证安全,在编译时报错
@ServiceProvider({ serviceName: BaseConstant.CLASS_SERVICE, singleton: true })
export class CustomService implements IRecordService, IServiceProvider {
doRecord(): void {
}
}
@ServiceProvider 参数释义
serviceName: 服务名 【必传】。
服务的唯一标识。如果重复,在编译期会直接报错。
singleton: 默认 false【可选】。
服务提供方提供出的服务是否为单例。
<br>
五、动态化能力 Action 的设计
Action 本质是一个全局的系统回调,主要用于预埋的一系列操作,例如:弹窗、上传日志、清理缓存。
与 Android 系统自带的广播通知类似,你可以在任何地方声明动作与处理方式。并且所有 Action 都是可以被跟踪的,只要你愿意,可以在日志中将所有的动作调用栈输出,以方便调试使用,这样在一定程度上可以解决观察者模式带来的通病:无法追踪 Observable 的问题。
<br>
5.1 Action 使用
声明一个 Action:
// action建议遵循一定的格式
const readonly ACTION = "therouter://action/xxx"
// action既可以放在ServiceProvider里面,也可以单独放在任意类中,但不能是top-level函数,这一点与Android不同
// action函数允许有返回值,可以做耗时操作
// 多次添加重复action,每个action的方法都会执行,但最终只会返回优先级最高的方法的返回值
@Action({ action: ACTION })
public test(par: string): string {
return "返回入参:" + par;
}
执行一个 Action:
// action建议遵循一定的格式
const val ACTION = "therouter://action/xxx"
// 如果执行了一个没有被声明的Action,则不会有任何动作
// 这里的"hello"字符串,是根据action定义时有一个入参,所以调用时需要传入这个参数
TheRouter.action<string>(ACTION, "hello").then((str) => {
this.text = str;
})
@Action 参数释义
action: 事件名 【必传】。
当前函数需要响应的 action。如果同一个 action 有多个函数订阅,会在响应时根据优先级决定先后顺序。
priority: 优先级 (number 类型),默认 5【可选】。
数字越大,优先级越高。
<br>
5.2 客户端动态响应使用场景
如果仅客户端使用,常用的场景可能是:当用户执行某些操作(打开某个页面、H5 点击某个按钮、动态页面配置的点击事件)时,将会自动触发,执行预埋的 Action 逻辑。
如果与服务端链路打通,这个能力其实是需要整个公司的配合,比如有一套类似智慧大脑的方案,可以基于客户端过去的一些埋点数据,智能推断出用户下一步要做的事情,然后通过长连接直接向客户端下发指令做某些事情。那么通过客户端预埋的页面跳转、弹窗、清缓存、退出登录等等操作,就可以通过服务端指令进行操作,则就是一套完整的动态化方案。
<img src="https://oscimg.oschina.net/oscnet//79f8e6cef031e34cfa80d4f329003662.png" class="blog-img">
<br>
六、从其他路由迁移至 TheRouter
6.1 迁移工具一键迁移?
TheRouter 在 Android 项目上提供了图形化界面的迁移工具,可以一键从其他路由迁移到 TheRouter。
鸿蒙当然也提供了相同的能力,你可以在插件市场搜索,直接安装。安装好以后,点击 IDE 顶部的 Tool 菜单,找到迁移工具。
七、总结
<br>
TheRouter 并不仅仅是一个小巧灵活的路由库,而是一整套 Android、iOS、Harmony 三端完整的移动端解决方案,对移动端开发者更友好,上手开发适应性更强。使用 TheRouter 能够解决几乎全部的模块化过程中会遇到的问题。
对于现有的路由框架,我们也在最大限度支持平滑迁移。你也可以在 Github issue 中提出需求,我们评估后会尽快支持,也欢迎任何人提供 Pull Requests。
<br>
更多问题请加群沟通:
<br> <img src="https://oscimg.oschina.net/oscnet//58894fe2e65215c645807394c7428b38.png" class="blog-img">
<br>
</string></irecordservice></string,>