【Java】利用反射执行Spring容器Bean指定的方法,支持多种参数自动调用
使用情景
- 将定时任务录入数据库(这样做的好处是定时任务可视化,也可以动态修改各个任务的执行时间),通过反射执行对应的方法;
- 配合Netty实现简单的HTTP请求处理框架
- 其他需要使用反射执行Spring方法的业务亦可
目的
很多文章都提到了反射,但是对于方法参数处理这一块都是明确了类型,不支持按照实际参数动态转换,而本篇文章提供了一个思路怎么做到方法参数的动态调用。
大家也可以通过利用本文的方法结合自己的业务场景写出复用性更高、可扩展性更好的代码。欢迎各位指出文章中的错误,如果有更好的思路可以在下方评论,我们一起讨论。
欢迎转发,请注明出处。
实现方式
前提:
明确清楚需要执行的类和方法。
思路
- 通过Spring容器获取需要执行的类,注意:从spring容器中获取的类可能是被JDK或CGLIB代理的(取决于你的环境配置);
- 获取执行的Mehod对象;
- 封装方法实际参数List,仅支持基本类型包装类, String,对象,Map等参数类型自动转换
- 执行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);
}
}
源码分享
【Java】利用反射执行Spring容器Bean指定的方法,支持多种参数自动调用的更多相关文章
- JAVA面试题:Spring中bean的生命周期
Spring 中bean 的生命周期短暂吗? 在spring中,从BeanFactory或ApplicationContext取得的实例为Singleton,也就是预设为每一个Bean的别名只能维持一 ...
- 基于反射启动Spring容器
基于反射启动Spring容器 package com.maple.test; import org.springframework.context.ApplicationContext; import ...
- java利用反射调用类的某个方法
java利用反射机制 可以动态调用某个类的某个方法,在 扩展系统功能或提供对外接口时经常用的到. 代码如下: 打印类Print.java package com.test.reflct; /** * ...
- java利用反射获取类的属性及类型
java利用反射获取类的属性及类型. import java.lang.reflect.Field; import java.math.BigDecimal; import java.util.Map ...
- 获取Spring容器Bean对象工具类
在开发中,总是能碰到用注解注入不了Spring容器里面bean对象的问题.为了解决这个问题,我们需要一个工具类来直接获取Spring容器中的bean.因此就写了这个工具类,在此记录一下,方便后续查阅. ...
- 【转】Java利用反射机制访问私有化构造器
Java利用反射机制访问私有化构造器 博客分类: java 我们都知道,当一个类的构造方法被设为私有的时候(private),在其他类中是无法用new来实例化一个对象的. 但是有一种方法可以把带有 ...
- java利用反射机制判断对象的属性是否为空以及获取和设置该属性的值
1.java利用反射机制判断对象的属性是否为空: Map<String,String> validateMap = new LinkedHashMap<String, String& ...
- java利用反射访问类的私有(private)属性及方法
Java语言中,在一个类中,为了不让外界访问到有的属性和方法,通常将其设置为private,用正常的方式(对象名.属性名,对象名.方法名)将无法访问此属性与方法,但有没有其他方法可以访问呢?答案是有的 ...
- DataTable和DataRow利用反射直接转换为Model对象的扩展方法类
DataTable和DataRow利用反射直接转换为Model对象的扩展方法类 /// <summary> /// 类 说 明:给DataTable和DataRow扩展方法,直接转换为 ...
随机推荐
- Fabric进阶(三)—— 使用SDK动态增加组织
在fabric网络运行过程中动态追加新的组织是相当复杂的,网上的资料也十分匮乏,大多是基于first-network这样的简单示例,而且是使用启动cli容器的方法来增加组织,几乎没有针对实际应用的解决 ...
- 技术大佬:我去,你竟然还不会用 this 关键字
上一篇文章写的是 Spring Boot 的入门,结果有读者留言说,Java 都还没搞完,搞什么 Spring Boot,唬得我一愣一愣的.那这篇就继续来搞 Java,推出广受好评的我去系列第四集:你 ...
- kali中安装漏洞靶场Vulhub(超详细)
前言 我们都知道,在学习网络安全的过程中,搭建漏洞靶场有着至关重要的作用.复现各种漏洞,能更好的理解漏洞产生的原因,提高自己的学习能力.下面我在kali中演示如何详细安装漏洞靶场Vulhub. 什么是 ...
- 一次 nginx 返回 302 问题解决
1.问题描述: 应用服务器通过post方式向nginx服务器发送http请求,返回 302 2.问题解决过程 2.1.查询nginx日志,开始以为302错误会在nginx的错误日志error.log, ...
- 读Pyqt4教程,带你入门Pyqt4 _010
在PyQt4教程的这部分中,我们讨论拖放操作. 拖放(Drag-and-drop)指的是图形用户界面(Graphical user interface)中,在一个虚拟的对象上按着鼠标键将之拖曳到另一个 ...
- java的Interger自动包装带来的问题
1 首先看一下以下代码: Integer b=7; Integer c=7; Integer r=234; Integer d=234; System.out.println(b==c); Syste ...
- PHP获取临时文件的目录路径
PHP获得临时文件的文件目录相对路径,能够 根据tempnam()和sys_get_temp_dir()函数来完成. 下边我们运用简单的代码实例,给大伙儿介绍PHP获得临时文件的文件目录相对路径的方式 ...
- Chisel3 - Tutorial - ShiftRegister
https://mp.weixin.qq.com/s/LKiXUgSnt3DzgFLa9zLCmQ 简单的寄存器在时钟的驱动下,逐个往下传值. 参考链接: https://github.com ...
- Java实现蓝桥杯十六进制转八进制
基础练习 十六进制转八进制 时间限制:1.0s 内存限制:512.0MB 提交此题 锦囊1 锦囊2 问题描述 给定n个十六进制正整数,输出它们对应的八进制数. 输入格式 输入的第一行为一个正整数n ( ...
- Java实现 蓝桥杯 数独游戏
你一定听说过"数独"游戏. 如图,玩家需要根据9×9盘面上的已知数字,推理出所有剩余空格的数字,并满足每一行.每一列.每一个同色九宫内的数字均含1-9,不重复. 数独的答案都是唯一 ...