Java(Spring) 通过反射获取对象实例导致@Autowired注入失效

问题描述:
在代码中使用反射获取对象实例后,对象实例中通过@Autowrite注解注入的属性值为null(注入失败),导致带反射获取的对象实例调用方法时出现空指针等情况。
问题原因:
通过反射获取对象实例相当于“new”了一个对象,所以这个对象并没有被Spring所管理,导致无法获取注入的属性。

1. 示例代码

// 消费者 使用newInstance获取对象实例
public class Consumer {

    private static final String CLASS_PATH = "com.example.demo.dome.Provider";

    public void fun1() throws Exception {
        Class<?> sourceClass = Class.forName(CLASS_PATH);
        Provider provider= (Provider) sourceClass.newInstance();
        provider.say();
    }
}
// 提供者
@Component
public class Provider {
    @Autowired
    Depend depend;

    public void say() {
        depend.say();
    }
}
// 依赖
@Component
public class Depend {
    public void say(){
        System.out.println("Get provider successful");
    }
}

2. 使用newInstance获取对象实例,DeBug查看反射结果及代码执行结果

Java(Spring) 通过反射获取对象实例导致@Autowired注入失效
Java(Spring) 通过反射获取对象实例导致@Autowired注入失效
无法获取到对象实例中注入的属性。

3. 解决方案

思路:因为反射获取的对象不被Spring所管理,所以可以从Spring中获取对象实例。

3.1 编写工具类

import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;

@Component
public class SpringContextUtil implements ApplicationContextAware {

    private static ApplicationContext applicationContext;

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        SpringContextUtil.applicationContext = applicationContext;
    }

    public static <T> T getBean(Class<T> clazz) {
        // 先判断是否存在上下文对象
        if (applicationContext == null) {
            return null;
        }
        return applicationContext.getBean(clazz);
    }

    public static <T> T getBean(String beanName,Class<T> clazz) {
        // 先判断是否存在上下文对象
        if (applicationContext == null) {
            return null;
        }
        return applicationContext.getBean(beanName,clazz);
    }

}

修改反射方法

// 消费者 通过Spring获取对象实例
public class Consumer {

    private static final String CLASS_PATH = "com.example.demo.dome.Provider";

    public void fun2() throws Exception {
        Class<?> sourceClass = Class.forName(CLASS_PATH);
        Provider provider= (Provider) SpringContextUtil.getBean(sourceClass);
        provider.say();
    }
}

3.2 使用从Spring中获取对象实例,DeBug查看反射结果及代码执行结果

Java(Spring) 通过反射获取对象实例导致@Autowired注入失效
Java(Spring) 通过反射获取对象实例导致@Autowired注入失效
至此通过反射无法获取对象注入属性的问题就解决了。

上一篇:SpringBoot手动获取Bean


下一篇:ApplicationContext是如何被注入的