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

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

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

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

// 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库。例如:

// pay module
@AutoService(IPayService.class)
public class PayServiceImpl implements IPayService {
    @Override


    public boolean verifyPayment(String orderId) {
        // 实现支付验证逻辑
        return true;
    }

}



// login module
@AutoService(ILoginService.class)
public class LoginServiceImpl implements ILoginService {
    @Override


    public User getUserInfo() {
        // 实现获取用户信息逻辑
        return new User("Tom", 18);
    }

}



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

// 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)获取到对应接口的实现类对象,并调用其方法进行通信或功能执行。例如:

// 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老皮!!!欢迎大家来找我探讨交流?

© 版权声明
THE END
喜欢就支持一下吧
点赞0

Warning: mysqli_query(): (HY000/3): Error writing file '/tmp/MYTe9HmI' (Errcode: 28 - No space left on device) in /www/wwwroot/583.cn/wp-includes/class-wpdb.php on line 2345
admin的头像-五八三
评论 抢沙发
头像
欢迎您留下宝贵的见解!
提交
头像

昵称

图形验证码
取消
昵称代码图片