使用情景

  1. 将定时任务录入数据库(这样做的好处是定时任务可视化,也可以动态修改各个任务的执行时间),通过反射执行对应的方法;
  2. 配合Netty实现简单的HTTP请求处理框架
  3. 其他需要使用反射执行Spring方法的业务亦可

目的

      很多文章都提到了反射,但是对于方法参数处理这一块都是明确了类型,不支持按照实际参数动态转换,而本篇文章提供了一个思路怎么做到方法参数的动态调用。

      大家也可以通过利用本文的方法结合自己的业务场景写出复用性更高、可扩展性更好的代码。欢迎各位指出文章中的错误,如果有更好的思路可以在下方评论,我们一起讨论。

      欢迎转发,请注明出处。

实现方式

前提:

明确清楚需要执行的类和方法。

思路

  1. 通过Spring容器获取需要执行的类,注意:从spring容器中获取的类可能是被JDK或CGLIB代理的(取决于你的环境配置);
  2. 获取执行的Mehod对象;
  3. 封装方法实际参数List,仅支持基本类型包装类, String,对象,Map等参数类型自动转换
  4. 执行Mehod的invoke方法

核心类

@Service
public class ReflectionService { @Resource
private ApplicationContext applicationContext; private static final List<Class> WRAP_CLASS = Arrays.asList(Integer.class, Boolean.class, Double.class,Byte.class,Short.class, Long.class, Float.class, Double.class, BigDecimal.class, String.class); /**
* 反射调用spring bean方法的入口
* @param classz 类名
* @param methodName 方法名
* @param paramMap 实际参数
* @throws Exception
*/
public void invokeService(String classz, String methodName, Map<String,Object> paramMap) throws Exception {
if(!applicationContext.containsBean(classz)) {
throw new RuntimeException("Spring找不到对应的Bean");
} // 从Spring中获取代理对象(可能被JDK或者CGLIB代理)
Object proxyObject = applicationContext.getBean(classz); // 获取代理对象执行的方法
Method method = getMethod(proxyObject.getClass(), methodName); // 获取代理对象中的目标对象
Class target = AopUtils.getTargetClass(proxyObject); // 获取目标对象的方法,为什么获取目标对象的方法:只有目标对象才能通过 DefaultParameterNameDiscoverer 获取参数的方法名,代理对象由于可能被JDK或CGLIB代理导致获取不到参数名
Method targetMethod = getMethod(target, methodName); if(method == null) {
throw new RuntimeException(String.format("没有找到%s方法", methodName));
} // 获取方法执行的参数
List<Object> objects = getMethodParamList(targetMethod, paramMap); // 执行方法
method.invoke(proxyObject, objects.toArray());
} /**
* 获取方法实际参数,不支持基本类型
* @param method
* @param paramMap
* @return
*/
private List<Object> getMethodParamList(Method method, Map<String, Object> paramMap) throws Exception {
List<Object> objectList = new ArrayList<>(); // 利用Spring提供的类获取方法形参名
DefaultParameterNameDiscoverer nameDiscoverer = new DefaultParameterNameDiscoverer();
String[] param = nameDiscoverer.getParameterNames(method); for (int i = 0; i < method.getParameterTypes().length; i++) {
Class<?> parameterType = method.getParameterTypes()[i]; Object object = null;
// 基本类型不支持,支持包装类
if(WRAP_CLASS.contains(parameterType)) {
if(param != null && paramMap.containsKey(param[i])){
object = paramMap.get(param[i]); object = ConvertUtils.convert(object, parameterType);
} }else if (!parameterType.isPrimitive() ) {
object = getInstance(parameterType); // 赋值
BeanUtils.populate(object, paramMap);
} objectList.add(object);
} return objectList;
} /**
* 获取类型实例
* @param parameterType
* @return
* @throws Exception
*/
private Object getInstance(Class<?> parameterType) throws Exception {
if(parameterType.isAssignableFrom(List.class)) {
return new ArrayList(); }else if(parameterType.isAssignableFrom(Map.class)) {
return new HashMap();
}else if(parameterType.isAssignableFrom(Set.class)) {
return new HashSet();
}
return parameterType.newInstance();
} /**
* 获取目标方法
* @param proxyObject
* @param methodStr
* @return
*/
private Method getMethod(Class proxyObject, String methodStr) {
Method[] methods = proxyObject.getMethods(); for(Method method : methods) {
if(method.getName().equalsIgnoreCase(methodStr)) {
return method;
}
} return null;
}
}

测试方法

package com.ywqonly.springtest.reflection;

import com.ywqonly.springtest.reflection.service.impl.ReflectionService;
import com.ywqonly.springtest.reflection.vo.CarVO;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner; import javax.annotation.Resource;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map; @RunWith(SpringRunner.class)
@SpringBootTest
public class SpringReflectionTest { @Resource
private ReflectionService reflectionService; @Test
public void paramTest() throws Exception {
Map<String, Object> paramMap = new HashMap<>(); paramMap.put("carName", "宝马");
paramMap.put("speed", "1");
reflectionService.invokeService("carServiceImpl", "start", paramMap);
} @Test
public void objectTest() throws Exception {
Map<String, Object> paramMap = new HashMap<>(); paramMap.put("carName", "宝马");
paramMap.put("speed", "2");
reflectionService.invokeService("carServiceImpl", "startByVO", paramMap);
} @Test
public void mapTest() throws Exception {
Map<String, Object> paramMap = new HashMap<>(); paramMap.put("carName", "宝马");
paramMap.put("speed", "3");
reflectionService.invokeService("carServiceImpl", "startByMap", paramMap);
} }

源码分享

GITHUB源码地址

【Java】利用反射执行Spring容器Bean指定的方法,支持多种参数自动调用的更多相关文章

  1. JAVA面试题:Spring中bean的生命周期

    Spring 中bean 的生命周期短暂吗? 在spring中,从BeanFactory或ApplicationContext取得的实例为Singleton,也就是预设为每一个Bean的别名只能维持一 ...

  2. 基于反射启动Spring容器

    基于反射启动Spring容器 package com.maple.test; import org.springframework.context.ApplicationContext; import ...

  3. java利用反射调用类的某个方法

    java利用反射机制 可以动态调用某个类的某个方法,在 扩展系统功能或提供对外接口时经常用的到. 代码如下: 打印类Print.java package com.test.reflct; /** * ...

  4. java利用反射获取类的属性及类型

    java利用反射获取类的属性及类型. import java.lang.reflect.Field; import java.math.BigDecimal; import java.util.Map ...

  5. 获取Spring容器Bean对象工具类

    在开发中,总是能碰到用注解注入不了Spring容器里面bean对象的问题.为了解决这个问题,我们需要一个工具类来直接获取Spring容器中的bean.因此就写了这个工具类,在此记录一下,方便后续查阅. ...

  6. 【转】Java利用反射机制访问私有化构造器

    Java利用反射机制访问私有化构造器 博客分类: java   我们都知道,当一个类的构造方法被设为私有的时候(private),在其他类中是无法用new来实例化一个对象的. 但是有一种方法可以把带有 ...

  7. java利用反射机制判断对象的属性是否为空以及获取和设置该属性的值

    1.java利用反射机制判断对象的属性是否为空: Map<String,String> validateMap = new LinkedHashMap<String, String& ...

  8. java利用反射访问类的私有(private)属性及方法

    Java语言中,在一个类中,为了不让外界访问到有的属性和方法,通常将其设置为private,用正常的方式(对象名.属性名,对象名.方法名)将无法访问此属性与方法,但有没有其他方法可以访问呢?答案是有的 ...

  9. DataTable和DataRow利用反射直接转换为Model对象的扩展方法类

    DataTable和DataRow利用反射直接转换为Model对象的扩展方法类   /// <summary> /// 类 说 明:给DataTable和DataRow扩展方法,直接转换为 ...

随机推荐

  1. Java并发包4--可重入锁ReentrantLock的实现原理

    前言 ReentrantLock是JUC提供的可重入锁的实现,用法上几乎等同于Synchronized,但是ReentrantLock在功能的丰富性上要比Synchronized要强大. 一.Reen ...

  2. 201771010128 王玉兰《面象对象程序设计 (Java) 》第六周学习总结

    ---恢复内容开始--- 第一部分:基础知识总结: 1.继承 A:用已有类来构建新类的一种机制,当定义了一个新类继承一个类时,这个新类就继承了这个类的方法和域以适应新的情况: B:特点:具有层次结构. ...

  3. OC 面向对象的特性

    面向对象的编程语言有封装.继承 .抽象.多态4个主要的特征. 面向对象编程有三大特性:封装.继承.多态.   1. 封装: 封装是保证软件部件具有优良的模块性的基础,封装的目标就是要实现软件部件的“高 ...

  4. Intel FPGA Clock Region概念以及用法

    目录 Intel FPGA 的Clock Region概念 Intel 不同系列FPGA 的Clock Region 1. Clock Region Assignments in Intel Stra ...

  5. Closures Basic

    Closures Closures are one of the most powerful features of JavaScript. JavaScript allows for the nes ...

  6. SpringBoot系列——状态机(附完整源码)

    1. 简单介绍状态机 2. 状态机的本质 3. 状态机应用场景 1. 简单介绍状态机 状态机由状态寄存器和组合逻辑电路构成,能够根据控制信号按照预先设定的状态进行状态转移,是协调相关信号动作.完成特定 ...

  7. sql注入讲解

    1.输入1' 发现数据库报错,原因是我们的输入直接被代入到数据库查询语句里面. 2.有没有办法可以不让他报错呢?可以尝试一下构造正确的数据库语法,使之不报错.比如输入 1 and 1=1 试试 sel ...

  8. Mysql基础(三)

    #DML语言 /* 数据操作语言 插入:insert insert into 表名(列名,...) values(值1,...); insert into 表名 set 列名=值, 列名=值,... ...

  9. lunix如何查看防火墙是否关闭和关闭开启防火墙命令

    查看防火墙是否关闭的命令如下: 1.通过 /etc/init.d/iptables status 或者 service iptables status命令 2.通过 iptables -L命令 查看 ...

  10. angularjs 路由切换回到顶部

    angularjs路由切换  页面不会回到顶部 问题: 在angularjs中  ui-sref或者$state.go(),通过路由切换页面,发现新打开的路由页面仍然停留在上一次的路由页面访问的位置. ...