Harmony 动态路由框架:TheRouter 开源

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)

一、为什么要使用 TheRouter

路由是现如今移动端开发中必不可少的功能,尤其是企业级APP,可以用于将多模块页面跳转的强依赖关系解耦,同时减少跨团队开发的互相依赖问题。

对于大型 APP 开发,基本都会选用模块化(或组件化)方式开发,对于模块间解耦要求更高。 TheRouter 是一整套完全面向模块化开发的解决方案,不仅能支持常规的模块依赖解耦、页面跳转,同时提供了模块化过程中常见问题的解决办法。

1.1 TheRouter 鸿蒙端的三大能力

Navigator:

  • 支持 Path 与页面多对一关系或一对一关系,可用于解决多端path统一问题
  • 页面 Path 支持正则表达式声明
  • 支持 json 格式路由表导出
  • 路由表支持为页面添加注释说明
  • 支持动态下发 json 路由表,降级任意页面为H5
  • 支持页面跳转拦截处理
  • 支持使用路由跳转到第三方 SDK 中的页面

ServiceProvider:

  • 支持跨模块依赖注入
  • 支持自定义注入项的创建规则,依赖注入可自定义参数
  • 支持注入对象缓存,多次注入,只会 new 一次对象

ActionManager:

  • 支持全局回调配置
  • 支持多对一链式响应
  • 支持优先级响应
  • 方法支持返回值与入参
  • 支持记录调用路径,解决调试期观察者模式无法追踪 Observable 的问题

二、鸿蒙路由方案

根据华为官方文档建议,从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,正常跳转到第三方页面。既然第三方都能跳转,那么其他的二方、或自己的独立模块自然也就可以正常跳转了。

三、使用 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'。
java 复制代码
@Route({ path: BaseConstant.MAIN_PAGE, description: 'Demo首页', params: ["hello", "路由表默认参数"], launchMode:'STANDARD' })
@Component
export struct MainPage {
}

3.2 发起页面跳转

传入的参数可以是 string 和基本数据类型、也可以是ESObject对象。

java 复制代码
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 完全相同,则认为是同一条路由,不会考虑参数是否相同

路由表生成规则:编译期按照如下顺序取并集

覆盖规则

根据如下顺序,如果相同,后者可以覆盖前者的路由表规则。

  1. 编译期解析注解生成路由表
  2. 首先取 业务模块(har/hap) 中的路由表
  3. 再取 主hsp module 代码中的路由表
  4. 最后取 resources/base/profile/RouteMap.json 文件中声明的路由表。
  • 如果编译期没有这个文件,会生成一份默认路由表放在这个目录内(编译完成后如果没有配置保留,会自动删掉);如果有,会将路由表合并。
  1. 运行时线上动态下发的路由表
  • 路由表允许线上动态下发,将覆盖本地路由表,详见 【3.4 动态路由表的设计与使用】

如果编译期没有这个文件,会生成一份默认路由表放在这个目录内(编译完成后如果没有配置保留,会自动删掉);如果有,会将路由表合并,因此,对于没办法修改代码的第三方SDK内部,如果希望通过路由打开,只需要手动在 RouteMap.json 文件中声明,就能通过路由打开了。

3.4 动态路由表的设计与使用

TheRouter 的路由表是动态添加的,项目每次编译后,会在 app 内生成一份当前模块的全量路由表。这个路由表也可以后续通过远程下发的方式使用,例如远端可以针对不同的APP版本,下发不同的路由表达到配置目的。这样如果将来线上某些页面发生Crash,可以通过将这个页面的落地页替换为H5的方式,临时解决这类问题。

有两种推荐的远程下发方式可供使用方选择:

  1. 将打包系统与配置系统打通,每次新版本打包后自动将所有模块(hsp、har、hap) resource/rawfile/ 目录中的路由表文件上传到配置系统,聚合成一个 json 后,下发给对应版本 APP 。优点在于全自动不会出错。
  2. 配置系统无法打通,线上手动下发需要修改的路由项,因为 TheRouter 会自动用最新下发的路由项覆盖包内的路由项。优点在于精确,且流量资源占用小。

动态路由限制 :准确的说,应该是鸿蒙系统的限制。在鸿蒙上,路由表必须是静态的并且在编译期确定下来。TheRouter Harmony 做了一些黑科技处理,允许动态加载一个或多个路由表,但是动态加载的路由页面必须是在编译期就已经存在的,不能凭空新增(类似 AndroidActivity,在编译后就不能再改或新增注册清单文件了)。

java 复制代码
// 与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 文档 第三部分】。

java 复制代码
// 所有拦截器方法均在 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同时支持更多页面跳转能力:

  • 为第三方库里面的页面添加路由表,达到对某些页面降级替换的目的;
  • 跳转过程拦截器(总共四层,可根据实际需求使用);
  • 跳转结果回调;

四、跨模块依赖注入 ServiceProvider 的设计

对于模块化开发中跨模块的调用,我们推荐采用 SOA(面向服务架构) 的设计方式,服务调用方与使用方完全隔离,调用模块外的能力不需要关注能力的提供者是谁。
ServiceProvider 的核心设计思想也是这样的,目前服务间的调用协议采用接口的方式。当然,也可以兼容不通过接口下沉而是直接调用的情况。

  • 服务提供方负责提供服务,不需要关心调用方是谁会在何时调用自己。
  • 服务的使用方只关注服务本身,不需要关心这个服务是谁提供的,只需要只能服务能提供哪些能力即可。

例如上面的图片:拉拉需要使用录音的服务,小货则向外提供一个录音的服务,由TheRouterServiceProvider负责撮合。

4.1 服务使用方:拉拉

她无需关心,IRecordService 这个接口服务是谁提供的,他只需要知道自己需要使用这样的一个服务就行了。

注:如果没有提供服务的提供方,TheRouter.get() 可能返回 undefined

java 复制代码
TheRouter.get<IRecordService>(BaseConstant.CLASS_SERVICE)?.doRecord()

4.2 服务提供方:小货

服务提供方需要声明一个提供服务的方法,用 @ServiceProvider 注解标记,并需要实现接口 IServiceProvider

java 复制代码
// 类名不限定,任意名字都行
// 所有的 ServiceProvider 必须实现 IServiceProvider 接口
// 多次添加重复serviceName,框架会保证安全,在编译时报错
@ServiceProvider({ serviceName: BaseConstant.CLASS_SERVICE, singleton: true })
export class CustomService implements IRecordService, IServiceProvider {

  doRecord(): void {
  }
}

@ServiceProvider 参数释义

  • serviceName : 服务名 【必传】。
    服务的唯一标识。如果重复,在编译期会直接报错。
  • singleton : 默认false【可选】。
    服务提供方提供出的服务是否为单例。

五、动态化能力 Action 的设计

Action 本质是一个全局的系统回调,主要用于预埋的一系列操作,例如:弹窗、上传日志、清理缓存。

与 Android 系统自带的广播通知类似,你可以在任何地方声明动作与处理方式。并且所有 Action 都是可以被跟踪的,只要你愿意,可以在日志中将所有的动作调用栈输出,以方便调试使用,这样在一定程度上可以解决观察者模式带来的通病:无法追踪 Observable 的问题

5.1 Action 使用

声明一个 Action:

java 复制代码
// 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:

java 复制代码
// 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【可选】。
    数字越大,优先级越高。

5.2 客户端动态响应使用场景

如果仅客户端使用,常用的场景可能是:当用户执行某些操作(打开某个页面、H5点击某个按钮、动态页面配置的点击事件)时,将会自动触发,执行预埋的 Action 逻辑。

如果与服务端链路打通,这个能力其实是需要整个公司的配合,比如有一套类似智慧大脑的方案,可以基于客户端过去的一些埋点数据,智能推断出用户下一步要做的事情,然后通过长连接直接向客户端下发指令做某些事情。那么通过客户端预埋的页面跳转、弹窗、清缓存、退出登录等等操作,就可以通过服务端指令进行操作,则就是一套完整的动态化方案。

六、从其他路由迁移至 TheRouter

6.1 迁移工具一键迁移?

TheRouter 在 Android 项目上提供了图形化界面的迁移工具,可以一键从其他路由迁移到TheRouter

鸿蒙当然也提供了相同的能力,你可以在插件市场搜索,直接安装。安装好以后,点击 IDE 顶部的 Tool 菜单,找到迁移工具。

6.2 与其他路由对比

功能 TheRouter HMRouter Navigation
具备三端高一致性 ✔️ ✖️ ✖️
注解生成路由表 ✔️ ✔️ ✖️
路由path支持正则表达式 ✔️ ✔️ ✖️
指定拦截器 ✔️(四大拦截器可根据业务定制) ✔️ ✖️
导出路由表 ✔️(路由文档支持添加注释描述) ✔️ ✖️
支持跨模块调用 ✔️ ✔️ ✖️
动态修改路由信息 ✔️ ✖️(未提供功能接口) ✔️(限制高,需提前定义,通过if/else修改实现)
远端路由表下发 ✔️ ✖️ ✖️
多 Path 对应同一页面(低成本实现双端path统一) ✔️ ✖️ ✖️
支持使用路由打开第三方SDK页面 ✔️ ✖️ ✖️

七、总结

TheRouter 并不仅仅是一个小巧灵活的路由库,而是一整套 AndroidiOSHarmony 三端完整的移动端解决方案,对移动端开发者更友好,上手开发适应性更强。使用 TheRouter 能够解决几乎全部的模块化过程中会遇到的问题。

对于现有的路由框架,我们也在最大限度支持平滑迁移。你也可以在 Github issue 中提出需求,我们评估后会尽快支持,也欢迎任何人提供 Pull Requests

更多问题请加群沟通: