TheRouter 的跨模块依赖注入实现原理

逻辑星语
• 阅读 855

TheRouter用于跨模块通信设计的ServiceProvider,核心设计思想是参考了SOA(面向服务架构)的设计方式。

具体到 Android 侧就是 AIDL 类似的实现。

模块化能力支持项:

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

1.0 依赖注入

用于跨模块通信使用,核心设计思想是参考了SOA(面向服务架构) 的设计方式。
具体到 Android 侧就是 AIDL 类似的实现:
例如当前有两个模块:A订单模块、B登录模块,下单需要获取用户信息。
这个业务场景就是,A需要使用获取用户信息的服务,B需要向外提供一个获取用户信息的服务。

TheRouter 的跨模块依赖注入实现原理

1.1 声明接口服务

首先声明一个接口,放入公共依赖层

// 假设当前有一个用户信息获取服务
public interface IUserService {
    String getUserInfo();
}

1.2 服务使用方

也就是上面例子的 A订单模块,他需要使用获取用户信息的服务
A无需关心,IUserService这个接口服务是谁提供的,他只需要知道自己需要使用这样的一个服务就行了。
如果没有提供服务的提供方,TheRouter.get()可能返回null

TheRouter.get(IUserService::class.java)?.getUserInfo()

1.3 服务提供方

服务提供方需要声明一个提供服务的方法,用@ServiceProvider注解标记。

  • 如果是 java,必须是 public static 修饰
  • 如果是 kotlin,建议写成 top level 的函数
  • 方法名不限
/**
 * 方法名不限定,任意名字都行
 * 返回值必须是服务接口名,如果是实现了服务的子类,需要加上returnType限定(例如下面代码)
 * 方法必须加上 public static 修饰,否则编译期就会报错
 */
@ServiceProvider
public static IUserService test() {
    return new IUserService() {
        @Override
        public String getUserInfo() {
            return "返回用户信息";
        }
    };
}

// 也可以直接返回对象,然后标注这个方法的服名是什么
@ServiceProvider(returnType = IUserService.class)
public static UserServiceImpl test() {
    xxx
}

2.0 自定义服务拦截器

使用场景:单模块调试时,可能会有需要 mock 其他模块提供的服务,TheRouter 允许自定义其他模块的实现。

Interceptor interceptor = new Interceptor() {
    @Override
    public <T> T interception(Class<T> clazz, Object... params) {
        if (clazz == IUserService.class) {
            return new IUserService();
        }
        return null;
    }
};
TheRouter.getRouterInject().addInterceptor(interceptor);

3.0 服务缓存

服务提供方运行对提供的服务做配置,对于无状态的服务,尽可能使用缓存方式减少对象创建次数,而有状态的服务,则每次创建新对象保证多次调用有不会互相污染状态(例如订单状态管理、商品销售状态等服务)。

声明服务缓存只需要在 @ServiceProvider的方法上新增额外注解即可,例如下面示例代码:

  • Singleton 表示这个对象会被永久缓存,对外部调用方而言相当于这个对象声明成了单例
  • NewInstance 表示每次都会返回新对象

如果两个注解同时被添加,则只有Singleton会生效。


// 注:如果都不加,默认是LRU+软引用缓存
// 如果两个注解同时被添加,则只有Singleton会生效。
@Singleton // 对外部调用方而言相当于这个对象声明成了单例
@NewInstance  // 每次都会返回新对象
public interface IUserService {
    public String getUserInfo();
}


@ServiceProvider
public static IUserService test() {
    return new IUserService() {
        @Override
        public String getUserInfo() {
            return "返回用户信息";
        }
    };
}
点赞
收藏
评论区
推荐文章
Wesley13 Wesley13
3年前
SOA面向服务的分布式架构详解
导语:SOA作为一种面向服务的架构,是一种软件架构设计的模型和方法论。从业务角度来看,一切以最大化“服务”的价值为出发点,SOA利用企业现有的各种软件体系,重新整合并构建起一套新的软件架构。这套软件架构能够随着业务的变化,随时灵活地结合现有服务,组成新软件,共同服务于整个企业的业务体系。简单的理解,我们可以把SOA看作是模块化的组件,每个模块都可以实现独立功
从一个 10 年程序员的角度告诉你:搞懂 Java 面向对象有多容易?
前言:1)java面向对象语言,面向过程围绕过程(解决问题步骤),面向对象围绕实体(名词,特性(属性),行为(动作、方法))。它们设计思想区别在于关心核心不同的。主流都是面向对象的。实际开发,先按面向对象思想进行设计,具体实现时面向过程(人习惯)2)java怎么支持面向对象呢?a.万物皆对象,所有的类都是Object子类b.java中支
Easter79 Easter79
3年前
springcloud分布式架构思想简析
一、springcloud分布式架构1、分布式架构是基于分层开发思想来进行实现对每一个模块进行依赖和统一功能,有提供API,提供者和消费者等父工程用来统一所有微服务所需要的依赖版本,不做实际依赖2、各个模块,微服务思考:api模块:用来存放实体类,所以在创建数据库中对应的实体类之后,该服务就完成
Wesley13 Wesley13
3年前
.net core 面试题
第1题,什么是ASPnetcore?首先ASPnetcore不是aspnet的升级版本。它遵循了dotnet的标准架构,可以运行于多个操作系统上。它更快,更容易配置,更加模块化,可扩展性更强。第2题,aspdotcore有哪些好的功能?第一是依赖注入。第二是日志系统架构。第三是引入了一个跨平台的网络服务器,kestre
Stella981 Stella981
3年前
ES6 模块化(Module)export和import详解
ES6模块化(Module)export和import详解ES6在语言标准的层面上,实现了模块功能,而且实现得相当简单,完全可以取代现有的CommonJS和AMD规范,成为浏览器和服务器通用的模块解决方案。ES6模块的设计思想,是尽量的静态化,使得编译时就能确定模块的依赖关系,以及输入和输出的变量。CommonJS和AMD模块,都只能
Stella981 Stella981
3年前
Android模块化初体验及教程(二)
上一章初级体验了一下模块化,但是还是有很多需要优化和改进的地方思考点:怎么单独运行某一个子模块,不影响到其他模块之间的依赖?子模块需要Application,难道要每个模块写一个?因为是同一个服务器,所以网络请求库,怎么跨模块间公用?模块间怎么通信?解决将项
Stella981 Stella981
3年前
Android依赖注入应用
依赖注入(DI)是一种设计模式,允许在运行时或编译时移除或改变硬编码的依赖性。使用依赖注入库可以减少编码量、把精力专注在更有价值的地方、降低维护成本。Android程序通常使用注解(Annotation,例如@Click)实现“声明式编程”和依赖注入。注:“声明式编程”告诉机器在什么地方做什么事(Where
Wesley13 Wesley13
3年前
Java 面向对象的设计原则
一、1、面向对象思想的核心:封装、继承、多态。2、面向对象编程的追求:  高内聚低耦合的解决方案;  代码的模块化设计;3、什么是设计模式:  针对反复出现的问题的经典解决方案,是对特定条件下(上下文)问题的设计方案的经验总结,是前人设计实践经验的精华。4、面向对象设计原则
Stella981 Stella981
3年前
Android 主Module引用依赖Module,却无法使用里面的依赖库
如果模块化开发中遇到多模块的AndroidManifest.xml没有合并or多模块的资源文件没有合并or模块Ainclude了模块B,而无法使用模块B内依赖的其他aar包中的类的时候or提示Support包版本不一致这篇文章可能就是你要的解决方案~举个栗子:比如我们现在有一个App模块设计为:主
融云IM即时通讯 融云IM即时通讯
8个月前
融云 IM 干货丨私有云IMKit的自定义功能是否支持跨平台使用?
私有云IMKit的自定义功能支持跨平台使用,具体如下:1.跨平台支持IMKit支持在多个主流平台上集成使用,包括Android、iOS、Web等。这意味着开发者可以使用一套代码,创建适用于多个平台的应用,提高开发效率和应用的可移植性。2.平台兼容性Andr
码农较瘦 码农较瘦
4个月前
鸿蒙Ability对比Android的Fragment
鸿蒙通过Ability和动态组件(NodeController)实现类似AndroidFragment的模块化UI,采用声明式开发,生命周期更简洁(onStart/Active/Inactive/Stop)。相比Fragment,鸿蒙在资源占用、渲染效率(支持Vulkan)、跨设备适配及权限管理方面更优,支持多设备无缝流转。Android依赖FragmentTransaction,生态成熟但内存消耗较高。鸿蒙系统以轻量化、高效能见长,兼顾开发效率与多终端适配。