一、@autowired 与new
new出来的对象无法调用@Autowired注入的Spring Bean,否则报空指针异常,
@Autowired注入Spring Bean,则当前类必须也是Spring Bean才能调用它,不能用new xxx()来获得对象,这种方式获得的对象无法调用@Autowired注入的Bean。
二、@autowired 与静态方法
1.spring框架应用中有些静态方法需要依赖被容器管理的类,像这样
public class CeErrorAlarm{
@autowired
private static CeAlarmDao ceAlarmDao;
public static ceErrorAlarm(){
ceAlarmDao.insert();
}
}
这样一定会报java.lang.NullPointerException: null异常。
2.原理剖析
静态变量、类变量不是对象属性,而是一个类的属性,所以静态方法是属于类(class)的,普通方法才是属于实体对象的(new出来的对象),spring注入是在容器中实例化对象,所以不能使用静态方法。
而使用静态变量、类变量扩大了静态方法的使用范围。
静态方法在spring中是不推荐使用的,依赖注入的主要目的是让容器去产生一个对象的实例,然后交给spring容器管理,在整个生命周期中使用他们,更加方便灵活
一旦你使用静态方法,就不再需要去产生这个类的实例,这会让testing变得更加困难,同时你也不能为一个给定的类,依靠注入方式去产生多个具有不同的依赖环境的实例,这种static field是隐含共享的,并且是一种global全局状态,spring同样不推荐这样去做。
3.解决办法
1.将@autowired注解加在构造方法上
public class CeErrorAlarm{
private static CeAlarmDao ceAlarmDao;
@autowired
public CeErrorAlarm(CeAlarmDao ceAlarmDao){
CeErrorAlarm.ceAlarmDao=ceAlarmDao;//将利用构造方法自动注入的对象赋值给static
}
public static ceErrorAlarm(){
ceAlarmDao.insert();
}
}
2.用@PostConstruct注解
public class CeErrorAlarm{
private static CeAlarmDao ceAlarmDao;
@Autowired
private CeAlarmDao ceAlarmDao2;
@PostConstruct
public void init() {
ceAlarmDao=ceAlarmDao2; //原理类似
}
public static ceErrorAlarm(){
ceAlarmDao.insert();
}
}
@PostConstruct:被@PostConstruct修饰的方法会在服务器加载Servle的时候运行,并且只会被服务器执行一次。PostConstruct在构造函数之后执行,init()方法之前执行。(PreDestroy()方法在destroy()方法执行执行之后执行)