Android解耦(一)基于接口的module解耦

一、如何实现基于接口的module解耦

基于接口的module解耦可以通过以下几个步骤来实现:

1. 创建一个公共module(base module),并在其中定义所有需要被其他module调用或通信的接口(interface)。例如:

java 复制代码
// base module
public interface IPayService {
    // 支付验证方法
    boolean verifyPayment(String orderId);
}

public interface ILoginService {
    // 获取用户信息方法
    User getUserInfo();
}

2. 在各个子module中分别实现这些接口,并向公共module注册自己的实现类(implementation)。这里有两种注册方式:一种是使用Java SPI技术,在src/main/resources/META-INF/services目录下创建以接口全限定名为文件名的文本文件,并在其中写入对应实现类全限定名;另一种是使用编译时注解,在实现类上添加@AutoService(Interface.class)注解,并引入Google AutoService库。例如:

java 复制代码
// pay module
@AutoService(IPayService.class)
public class PayServiceImpl implements IPayService {
    @Override
    public boolean verifyPayment(String orderId) {
        // 实现支付验证逻辑
        return true;
    }
}
java 复制代码
// login module
@AutoService(ILoginService.class)
public class LoginServiceImpl implements ILoginService {
    @Override
    public User getUserInfo() {
        // 实现获取用户信息逻辑
        return new User("Tom", 18);
    }
}

3. 在公共module中创建一个组件管理类(ComponentManager),并在其中使用Java SPI技术加载所有已注册的组件接口和组件实现类之间对应关系,并提供一个静态方法来根据组件接口类型获取对应组件实现类对象。例如:

java 复制代码
// base module
public class ComponentManager {

    private static final Map<Class<?>, Object> COMPONENT_MAP = new HashMap<>();

    static {
        // 使用Java SPI技术加载所有已注册的组件接口和组件实现类之间对应关系
        ServiceLoader.load(Object.class).forEach(service -> {
            Class<?>[] interfaces = service.getClass().getInterfaces();
            if (interfaces != null && interfaces.length > 0) {
                for (Class<?> interfaceClass : interfaces) {
                    COMPONENT_MAP.put(interfaceClass, service);
                }
            }
        });
    }

    // 提供一个静态方法来根据组件接口类型获取对应组件实现类对象
    public static <T> T getComponent(Class<T> interfaceClass) {
        return (T) COMPONENT_MAP.get(interfaceClass);
    }
}

在其他module中,通过公共module的组件管理类(ComponentManager)获取到对应接口的实现类对象,并调用其方法进行通信或功能执行。例如:

java 复制代码
// social module
public class SocialActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_social);

        // 通过公共module的组件管理类(ComponentManager)获取到登录模块的接口实现类对象
        ILoginService loginService = ComponentManager.getComponent(ILoginService.class);

        // 调用登录模块的接口方法获取用户信息
        User user = loginService.getUserInfo();

        // 显示用户信息
        TextView tvUserName = findViewById(R.id.tv_user_name);
        TextView tvUserAge = findViewById(R.id.tv_user_age);
        tvUserName.setText(user.getName());
        tvUserAge.setText(String.valueOf(user.getAge()));
    }
}

二、基于接口的module解耦的优缺点

基于接口的module解耦有以下优点:

1. 解耦彻底,各个子module之间不需要有任何依赖或引用,只需要依赖公共module即可。
2. 通信简单高效,支持所有基本类型和类类型,像在调用module内部接口一样方便。
3. 实现灵活多样,可以根据不同场景或需求选择不同的接口实现类,并动态替换或切换。

基于接口的module解耦也有以下缺点:

1. 需要额外创建一个公共module,并在其中定义所有需要被其他module调用或通信的接口,增加了代码量和维护成本。
2. 需要使用Java SPI技术或编译时注解来注册组件接口和组件实现类之间对应关系,增加了编译时间和运行时开销。
3. 需要注意避免循环依赖或重复注册等问题,否则可能导致程序崩溃或异常。

三、基于接口的module解耦的使用场景

基于接口的module解耦适合以下使用场景:

1. 当项目功能较多且复杂时,需要将一个app主模块拆分成多个子模块(module),每个子模块负责一个独立的功能或业务场景。
2. 当子模块之间需要相互调用或通信时,需要一种方式来实现子模块之间的解耦和通信。

当子模块之间调用或通信涉及到多种数据类型时,需要一种方式来支持所有基本类型和类类型。
3. 当子模块之间调用或通信可能发生变化时,需要一种方式来实现灵活多样的切换或替换。

👀关注公众号:Android老皮!!!欢迎大家来找我探讨交流👀

相关推荐
沐言人生40 分钟前
React Native 源码分析1——HybridData 机制深度分析
android·react native
程序员陆业聪1 小时前
跨平台框架全景图:Flutter/KMP/KuiKly/RN的2026年格局
android
码云数智-园园2 小时前
Fibers(纤程)来了:打破阻塞,实现纯PHP下的异步非阻塞IO
android
shaoming37765 小时前
检查系统硬件配置是否满足PyCharm最低要求
android·spring boot·mysql
一起搞IT吧5 小时前
高通Camx功能feature分析之十五:insensor zoom介绍及实现
android·智能手机·相机
aqi007 小时前
一文读懂 HarmonyOS 6.1 带来的十大重要升级
android·华为·harmonyos·鸿蒙·harmony
秋99 小时前
MySQL 9.7.0 使用详解:新特性、实战与避坑指南
android·数据库·mysql
狼与自由9 小时前
clickhouse ReplacingMergeTree
android·clickhouse
吉吉619 小时前
php反序列化基础知识前奏
android·php·反序列化
努力努力再努力wz9 小时前
【MySQL进阶系列】拒绝冗余SQL:带你透彻理解视图的底层逻辑
android·c语言·数据结构·数据库·c++·sql·mysql