`TheRouter` 是一个用于移动端APP,包括 Android、iOS、Harmony 三端的模块化、组件化开发的一整套解决方案框架。提供了三端高一致性,对移动端开发者更友好,让开发人员更适应,使用起来也更顺手。在鸿蒙上, TheRouter 基于HMRouter做了深度定制,不仅支持平台化应用实现组件化、跨模块调用、动态化等功能的集成等功能基础上,还提供了编译时安全检查、支持动态路由下发与修改、路由 Path 一对多等高度动态能力。     

Github: [https://github.com/HuolalaTech/hll-wp-therouter-harmony/](https://github.com/HuolalaTech/hll-wp-therouter-harmony/)   
官网:[http://therouter.cn/](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](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'。  

<br>

```java
@Route({ path: BaseConstant.MAIN_PAGE, description: 'Demo首页', params: ["hello", "路由表默认参数"], launchMode:'STANDARD' })
@Component
export struct MainPage {
}
```

<br>

#### 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();
       
```

<br>

#### 3.3 路由表生成规则  

如果两条路由的 `path` 完全相同,则认为是同一条路由,**不会考虑参数是否相同**。   
路由表生成规则:编译期按照如下顺序取**并集**。   

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

1. 编译期解析注解生成路由表
2. 首先取 `业务模块(har/hap)` 中的路由表
3. 再取 主`hsp module` 代码中的路由表
4. 最后取 `resources/base/profile/RouteMap.json` 文件中声明的路由表。
  - 如果编译期没有这个文件,会生成一份默认路由表放在这个目录内(编译完成后如果没有配置保留,会自动删掉);如果有,会将路由表合并。   
5. 运行时线上动态下发的路由表  
  - 路由表允许线上动态下发,将覆盖本地路由表,详见 【3.4 动态路由表的设计与使用】  
 
如果编译期没有这个文件,会生成一份默认路由表放在这个目录内(编译完成后如果没有配置保留,会自动删掉);如果有,会将路由表合并,因此,对于没办法修改代码的第三方SDK内部,如果希望通过路由打开,只需要手动在 `RouteMap.json` 文件中声明,就能通过路由打开了。    

<br>

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

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

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

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

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

<br>

```java
// 与Android逻辑不同,此代码 必须 在页面打开之前,路由初始化之后调用。 建议紧跟 TheRouter.init() 调用
TheRouter.setRouteMapInitTask(task: (map: Map<string, RouteItem>) => void);

/** 
 * 此处的 map 就是当前应用的路由表全量,
 * 当获取到远端路由表以后,把路由表继续传入map中,有重复项可自动覆盖
 */
TheRouter.setRouteMapInitTask(() => {
    // 此处为纯业务逻辑,每家公司远端配置方案可能都不一样
    const json = Connfig.doHttp("routeMap");
    // 只需要将路由json返回给框架即可,不建议在任务中做耗时操作
    return json;
});
```

<br>

#### 3.5 拦截器用法

框架内置四种自定义处理器可供业务场景定制,用于在路由跳转过程中,以切面的方式统一修改路由落地页参数信息。   
Harmony 路由与 Android 路由的拦截器使用完全一致,可以直接参考【[Android 文档](https://therouter.cn/docs/2022/08/28/01) 第三部分】。

```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同时支持更多页面跳转能力:    

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

<br>

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

对于模块化开发中跨模块的调用,我们推荐采用 [SOA(面向服务架构)](https://zh.m.wikipedia.org/zh-cn/%E9%9D%A2%E5%90%91%E6%9C%8D%E5%8A%A1%E7%9A%84%E4%BD%93%E7%B3%BB%E7%BB%93%E6%9E%84) 的设计方式,服务调用方与使用方完全隔离,调用模块外的能力不需要关注能力的提供者是谁。  
`ServiceProvider` 的核心设计思想也是这样的,目前服务间的调用协议采用接口的方式。当然,也可以兼容不通过接口下沉而是直接调用的情况。  

<img src="https://cdn.kymjs.com:8843/qiniu/images/blog_image/therouter/3.jpeg" class="blog-img"/>

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

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

<br>

#### 4.1 服务使用方:拉拉

她无需关心,`IRecordService` 这个接口服务是谁提供的,他只需要知道自己需要使用这样的一个服务就行了。  
注:如果没有提供服务的提供方,`TheRouter.get()` 可能返回 `undefined`  

```java
TheRouter.get<IRecordService>(BaseConstant.CLASS_SERVICE)?.doRecord()
```

<br>

#### 4.2 服务提供方:小货  

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

```java

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

  doRecord(): void {
  }
}

```

<br>

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

<br>

### 五、动态化能力 Action 的设计   

`Action` 本质是一个全局的系统回调,主要用于预埋的一系列操作,例如:弹窗、上传日志、清理缓存。  
与 Android 系统自带的广播通知类似,你可以在任何地方声明动作与处理方式。并且所有 `Action` 都是可以被跟踪的,只要你愿意,可以在日志中将所有的动作调用栈输出,以方便调试使用,这样在一定程度上可以解决观察者模式带来的通病:**无法追踪 `Observable` 的问题**。

<br>

#### 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;
}

```

<br>

执行一个 Action:

```java
// action建议遵循一定的格式
const val ACTION = "therouter://action/xxx"

// 如果执行了一个没有被声明的Action,则不会有任何动作
// 这里的"hello"字符串,是根据action定义时有一个入参,所以调用时需要传入这个参数
TheRouter.action<string>(ACTION, "hello").then((str) => {
    this.text = str;
})
```

<br>

**@Action 参数释义**   
 
* **action**: 事件名 【必传】。   
 当前函数需要响应的action。如果同一个action有多个函数订阅,会在响应时根据优先级决定先后顺序。   
* **priority**: 优先级(number类型),默认5【可选】。  
 数字越大,优先级越高。   

<br>

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

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

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

<img src="https://cdn.kymjs.com:8843/qiniu/therouter/actionmanager.png" class="blog-img"/>

<br>

### 六、从其他路由迁移至 TheRouter

#### 6.1 迁移工具一键迁移?  

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

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

<br>

#### 6.2 与其他路由对比

<br>

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


<br>

### 七、总结

<br>

`TheRouter` 并不仅仅是一个小巧灵活的路由库,而是一整套 `Android`、`iOS`、`Harmony` 三端完整的移动端解决方案,对移动端开发者更友好,上手开发适应性更强。使用 `TheRouter` 能够解决几乎全部的模块化过程中会遇到的问题。  
对于现有的路由框架,我们也在最大限度支持平滑迁移。你也可以在 `Github` `issue` 中提出需求,我们评估后会尽快支持,也欢迎任何人提供 `Pull Requests`。  

<br>

更多问题请加群沟通:  
<br>
<img src="https://cdn.kymjs.com:8843/qiniu/therouter/therouter_wx_group.png" class="blog-img"/>

<br>

Logo

讨论HarmonyOS开发技术,专注于API与组件、DevEco Studio、测试、元服务和应用上架分发等。

更多推荐