springboot中 利用java反射调用Service,注入Dao接口为null

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("个性化表单数据插入失败");
            }
        }

你可能感兴趣的:(框架)