在开发组件化项目中,遇到一个这样的问题,两个不同的Module相互之间没有任何直接依赖关系,现在需求是需要在Module_A中调用Module_B中的某个类的方法,以下为解决此问题的方法;

采用的核心技术:路由、反射;

解决问题的过程,首先将需要调用的类通过路由注册,注册到路由中心,这里讲Module_B中的类注册到了路由中心,在Module_A中首先先通过路由中心获取到已经注册了的Module_B的类,获取到Class后通过反射原理获取当前对象实例,然后继续通过反射调用调用当前实例对象的对应方法(支持无参方法及有参方法),至此调用方法已实现,具体示例如下:

1、路由注册工具类:

/**
* Class类路由注册工具类
* Created by sun.li on 2018/7/6.
*
* @author sun.li
*/ public class RouterClassUtil implements RouterInter<Object,Class<? extends Object>>{ private ConcurrentHashMap<String, Class<? extends Object>> mClassZ = new ConcurrentHashMap();
private ConcurrentHashMap<String, Object> mClassA = new ConcurrentHashMap();
private static RouterClassUtil mInstance; private RouterClassUtil() {
} public static RouterClassUtil getInstance() {
if(null == mInstance) {
synchronized(RouterActivity.class) {
if(null == mInstance) {
mInstance = new RouterClassUtil();
}
}
}
return mInstance;
} @Override
public void routerT(String key, Object obj) {
Object cacheActivity = this.mClassA.get(key);
if(null == cacheActivity || obj != cacheActivity) {
this.mClassA.put(key, obj);
}
} @Override
public Object invokT(String key) {
return this.mClassA.get(key);
} @Override
public void routerV(String key, Class<?> aClass) {
Class cacheClass = this.mClassZ.get(key);
if(null == cacheClass || aClass != cacheClass) {
this.mClassZ.put(key, aClass);
}
} @Override
public Class<? extends Object> invokV(String key) {
return this.mClassZ.get(key);
} /** 获取Object对象*/
public Object getObject(String key){
Object object = invokT(key);
if(null == object){
try {
Class<? extends Object> aClass = invokV(RouterHttpClassKey.KEY_MODULE_HTTP_HTTPUTIL);
if(null != aClass) {
object = aClass.newInstance();//通过Class对象实例化当前对象
}
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
}
return object;
}
} 2、路由对应Class配置类(此处声明具体的Class类路径,注册时通过反射原理获取对应的Class): /**
* 路由跳转Class对应key配置类
* Created by sun.li on 2018/7/6.
*
* @author sun.li
*/ public class RouterHttpClassKey { /**
* HttpUtil
*/
public static final String KEY_MODULE_HTTP_HTTPUTIL = "**.**.**.util.HttpLibUtil";
} 3、路由注册: RouterClassUtil.getInstance().routerV(value,correspondingClass); 4、路由调用获取当前的对象并且通过反射调用对应的方法: 4.1、无参方法调用: /** 调用Module中的对应无参方法*/
public void callMethodInModule(@NonNull String className,@NonNull String methodName){
Object object = RouterClassUtil.getInstance().getObject(className);
if (null != object) {
Method method = getMethod(object,methodName,null);
if(null!=method){
try {
//执行对象object中通过反射获取的方法
method.invoke(object);
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
}
}
} 4.2、有参方法调用(发射调用多参数时因为方法限制,还未想到好的方式): /**
* 公共网络请求参数对象
* Created by sun.li on 2018/7/6.
*
* @author sun.li
*/ public class BaseObjectParameterBean { public BaseObjectParameterBean(){ } public BaseObjectParameterBean(Class parameterType, Object parameterValue){
setParameterType(parameterType);
setParameterValue(parameterValue);
} /** 参数值*/
private Object parameterValue; /** 参数类型*/
private Class parameterType; public Object getParameterValue() {
return parameterValue;
} public void setParameterValue(Object parameterValue) {
this.parameterValue = parameterValue;
} public Class getParameterType() {
return parameterType;
} public void setParameterType(Class parameterType) {
this.parameterType = parameterType;
}
} /** 调用Module中的对应有参方法*/
public void callMethodInModule(@NonNull String className, @NonNull String methodName, List<BaseObjectParameterBean> mList){
if(null!=mList && mList.size()>0){
/* 参数类型集合*/
List<Class> classList = new ArrayList<>();
/* 参数值集合*/
List<Object> objectList = new ArrayList<>();
for (int i = 0; i < mList.size(); i++) {
BaseObjectParameterBean baseHttpRequestParameterBean = mList.get(i);
if(null != baseHttpRequestParameterBean){
if(null != baseHttpRequestParameterBean.getParameterValue() && null != baseHttpRequestParameterBean.getParameterType()){
classList.add(baseHttpRequestParameterBean.getParameterType());
objectList.add(baseHttpRequestParameterBean.getParameterValue());
}
}
}
if(classList.size()>0 && objectList.size()>0){
Object object = RouterClassUtil.getInstance().getObject(className);
if (null != object) {
Method method = getMethod(object,methodName,classList);
if(null!=method){
try {
//执行对象object中通过反射获取的方法
Object[] mObjectList = new Object[objectList.size()];
for (int i = 0; i < objectList.size(); i++) {
if(null != objectList.get(i)){
mObjectList[i] = objectList.get(i);
}
}
method.invoke(object,mObjectList);
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
}
}
}
}else{
callMethodInModule(className,methodName);
}
} private Method getMethod(Object object, @NonNull String methodName, List<Class> mList){
if(null == object || TextUtils.isEmpty(methodName)){
return null;
}
Method method = null;
try { if(null != mList && mList.size()>0){
// 父类对象调用子类有参数的方法(反射原理)
Class[] mClassList = new Class[mList.size()];
for (int i = 0; i < mList.size(); i++) {
if(null != mList.get(i)){
mClassList[i] = mList.get(i);
}
}
method = object.getClass().getMethod(methodName,mClassList);
}else{
method = object.getClass().getMethod(methodName);// 父类对象调用子类方法(反射原理)
}
} catch (NoSuchMethodException e) {
e.printStackTrace();
}
return method;
} 4.3、调用示例: BaseObjectUtil.getInstance().callMethodInModule(RouterHttpClassKey.KEY_MODULE_HTTP_HTTPUTIL,"test"); List<BaseHttpRequestParameterBean> beans = new ArrayList<>();
beans.add(new BaseHttpRequestParameterBean(String.class,"哈哈"));
beans.add(new BaseHttpRequestParameterBean(Integer.class,88)); BaseHttpRequestParameterBean baseHttpRequestParameterBean = new BaseHttpRequestParameterBean();
baseHttpRequestParameterBean.setParameterType(Integer.class);
beans.add(baseHttpRequestParameterBean); BaseObjectUtil.getInstance().callMethodInModule(RouterHttpClassKey.KEY_MODULE_HTTP_HTTPUTIL,"test2",beans); 调用结果: 07-06 16:07:29.862 10329-10329/cc.**.** E/HttpLibUtil:: 我是HttpLib中的test方法
07-06 16:07:29.862 10329-10329/cc.**.** E/HttpLibUtil:: 我是HttpLib中的test2方法,str=哈哈 i=88

java(Android)跨Module调用对应类方法需求解决方案的更多相关文章

  1. AllJoyn+Android开发案例-android跨设备调用方法

    AllJoyn+Android开发案例-android跨设备调用方法 项目须要涉及AllJoyn开源物联网框架.前面主要了解了一些AllJoyn主要的概念.像总线,总线附件,总线对象,总线接口这种概念 ...

  2. Android 跨应用调用Activity

    http://blog.csdn.net/ouyangliping/article/details/7972141 如何调用另外一个app应用的activity或者service,本文提供一个验证可行 ...

  3. Android 跨应用调用Activity及Service

    如何调用另外一个app应用的activity或者service,本文提供一个验证可行的方法. 调用方法: Intent intent=new Intent("youActionName&qu ...

  4. Android 跨进程调用忽略权限

    Framework层: @Override    public StackInfo getStackInfo(int stackId) {        final int callingUid = ...

  5. JAVA如何跨项目调用接口

    public String load(String url, String query) throws Exception { URL restURL = new URL(url); /* * 此处的 ...

  6. Spring AOP调用本类方法为什么没有生效

    首先请思考一下以下代码执行的结果: LogAop.java //声明一个AOP拦截service包下的所有方法@Aspectpublic class LogAop { @Around("ex ...

  7. Android中的跨进程调用技术AIDL

    什么是AIDL Android系统中的进程之间不能共享内存,因此,需要提供一些机制在不同进程之间进行数据通信. 为了使其他的应用程序也可以访问本应用程序提供的服务,Android系统采用了远程过程调用 ...

  8. Golang、Php、Python、Java基于Thrift0.9.1实现跨语言调用

    目录: 一.什么是Thrift? 1) Thrift内部框架一瞥 2) 支持的数据传输格式.数据传输方式和服务模型 3) Thrift IDL 二.Thrift的官方网站在哪里? 三.在哪里下载?需要 ...

  9. Android四大组件应用系列5——使用AIDL实现跨进程调用Service

    一.问题描述 Android应用程序的四大组件中Activity.BroadcastReceiver.ContentProvider.Service都可以进行跨进程.在上一篇我们通过ContentPr ...

随机推荐

  1. 小程序开通微信支付 --- 微信商户平台绑定微信小程序APPID

    首先情况是这样的:现有公司有个公众号,已经开通了微信支付(已经有一个商户平台),现在需要开发 微信小程序(也有微信支付),如果在小程序里面重新申请 微信支付,就显得比较麻烦.腾讯官方已经提供了 一个商 ...

  2. MUI事件管理

    模块:事件管理 http://dev.dcloud.net.cn/mui/event/ 事件绑定: 除了可以使用addEventListener()方法监听某个特定元素上的事件外, 也可以使用.on( ...

  3. Django - 项目总结

    总结: 基础,进阶,项目 Django - 练习(图书管理系统) 前后端分离得项目: vue + rest framework 项目: 图书增删改查页面 BBS + BLOG系统 CRM系统   在线 ...

  4. 调试:Spring AOP执行过程

    调试项目:https://github.com/1367356/laboratoryWeb 断点位置 点击查询:http://localhost:9002/queryNews?htmlid=15318 ...

  5. Django—Form两种保留用户提交数据的方法

    用户在网页上进行表单填写时,有可能出现某项填写错误.一般情况下,用户在未发觉错误的情况下点击提交,则此表单的内容会清空,用户不得不再重新填写,这样的用户体验是及其糟糕的. 在此,我们有2种方法将用户的 ...

  6. 模块讲解----configparser模块(my.cnf配置文件操作)

    查询 1.所有节点: 2.指定节点下的所有key和values: 3.指定节点下所有的key: 4.指定节点和key下的values: # #configparser用于处理特定格式的文件,其本质上是 ...

  7. SQL Server 一些使用小技巧

    1.查询的时候把某一个字段的值拼接成字符串 以下是演示数据. 第一种方式:使用自定义变量 ) SET @Names='' -- 需要先赋值为空字符串,不然结果会是 null SELECT @Names ...

  8. Mybatis使用generatedKey在插入数据时返回自增id始终为1,自增id实际返回到原对象当中的问题排查

    今天在使用数据库的时候,遇到一个场景,即在插入数据完成后需要返回此数据对应的自增主键id,但是在使用Mybatis中的generatedKey且确认各项配置均正确无误的情况下,每次插入成功后,返回的都 ...

  9. RabittMQ实践(一): RabbitMQ的安装、启动

    安装:   启动监控管理器:rabbitmq-plugins enable rabbitmq_management 关闭监控管理器:rabbitmq-plugins disable rabbitmq_ ...

  10. Terminal(终端) 在 OS X下如何快速调用

    Terminal(终端) 在 OS X下如何快速调用 转载请注明原作者:文章如果对您有所启发或帮助,不介意您请我喝一杯咖啡 ​ Terminal作为人机交流中极其重要的一部分,无论是在Windows. ...