一、如何实现基于接口的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、当子模块之间调用或通信可能发生变化时,需要一种方式来实现灵活多样的切换或替换。