1、反射简介
反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;
对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及
动态调用对象的方法的功能称为java语言的反射机制。
2、问题:
在项目中打算做一个通用的导出方法,但是这个方法是写在一个普通的工具类中的,这个工具类中我们通过使用反射的方法去调用其他的service层,通过service层插入数据库实体对象,但是serviceImpl中的dao接口对象却为空。经过调查由于使用反射,导致dao注入失败。原因是自动装配是在 spring环境下当使用该类的实例时由spring容器完成了类的实例化过程,当然包括对依赖对象的实例化过程而通过反射创建实例时,是根据你调用的构造函数完成的实例化过程,没有 容器的自动化创建实例了,所以需要自己对依赖对象进行注入。所以依赖spring容器实例化
和自己用反射实例化是两种独立的方式,不能相互渗透的。
3、代码解析
a:原本我们使用反射调用service层错误的方式
Class<?> classType = Class.forName(serviceClass);
Method m = classType.getDeclaredMethod("method名称",new Class[]{parameters.class});
List<?> list = m.invoke(classType.newInstance(),parameters);
注意:这里我们就是使用classType.newInstance()方法才会使service中的dao注入失败。
b:正确的方法,通过spring容器取得对象
WebApplicationContext wac = ContextLoader.getCurrentWebApplicationContext();
Class<?> classType = Class.forName(serviceClass);
Method m = classType.getDeclaredMethod("method名称",new Class[]{parameters.class});
List<?> list = m.invoke(wac.getBean("service的id对象名称"),parameters);
4、以上的代码是别人spring的代码,由于本人是用springboot,所以重新写了下代码。
(1)、工具类:SpringBootBeanUtil.java
import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;
/**
* SpringBoot 普通类获取Spring容器中的bean工具类
* @author lvgang
*/
@Component
public class SpringBootBeanUtil implements ApplicationContextAware {
private static ApplicationContext applicationContext;
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
if (SpringBootBeanUtil.applicationContext == null) {
SpringBootBeanUtil.applicationContext = applicationContext;
}
System.out.println("========ApplicationContext配置成功========");
System.out.println("========在普通类可以通过调用SpringBootBeanUtil.getApplicationContext()获取applicationContext对象========");
System.out.println("========applicationContext="+ SpringBootBeanUtil.applicationContext +"========");
}
/**
* 获取applicationContext
* @return
*/
public static ApplicationContext getApplicationContext() {
return applicationContext;
}
/**
* 通过name获取 Bean.
* @param name
* @return
*/
public static Object getBean(String name) {
return getApplicationContext().getBean(name);
}
/**
* 通过class获取Bean.
* @param clazz
* @return
*/
public static <T> T getBean(Class<T> clazz) {
return getApplicationContext().getBean(clazz);
}
/**
* 通过name,以及Clazz返回指定的Bean
* @param name
* @param clazz
* @return
*/
public static <T> T getBean(String name, Class<T> clazz) {
return getApplicationContext().getBean(name, clazz);
}
}
(2)调用:forName中的为自己获取并拼接的类地址
try {
//从ApplicationContext中取出已创建好的的对象
//不可直接反射创建serviceimpi对象,因为反射创建出来的对象无法实例化dao接口
ApplicationContext applicationContext = SpringBootBeanUtil.getApplicationContext();
//反射创建serviceimpi实体对象,和实体类
Class<?> ServiceImplType = Class.forName(GlobalParams.REF_SERVICE+className+"ServiceImpl");
Class<?> entityType = Class.forName(GlobalParams.REF_ENTITY+className);
//反射设置方法参数。
Method method = ServiceImplType.getDeclaredMethod("Insert",entityType);
//在ApplicationContext中根据class取出已实例化的bean
method.invoke(applicationContext.getBean(ServiceImplType),className);
} catch (ClassNotFoundException | IllegalAccessException | InvocationTargetException | NoSuchMethodException e) {
e.printStackTrace();
return GlobalResult.resOk("个性化表单数据插入失败");
}
}