zoukankan      html  css  js  c++  java
  • 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("个性化表单数据插入失败");
                }
            }
  • 相关阅读:
    线程数究竟设多少合理
    Elasticsearch 技术分析(八):剖析 Elasticsearch 的索引原理
    作为程序员你是如何学习的?
    系统运行缓慢,CPU 100%,以及Full GC次数过多问题的排查思路
    kms相关文档
    删除所有docker容器镜像
    ubuntu mysql5.7安装
    GORM自定义日志配置
    SQL清空全部表数据
    Nginx 证书
  • 原文地址:https://www.cnblogs.com/shuilangyizu/p/14189588.html
Copyright © 2011-2022 走看看