Spring中@Autowired注解和静态方法关联应用
背景:
static修饰的成员(包括变量和方法)叫静态成员,都存放在方法区中的静态区,随着类的加载而存在、类的回收而释放,和对象存在与否没有关系,并且要先于对象存在。(所以导致static修饰的方法,里面的东西不要依赖于其他实例化对象,依赖的话就会出现空指针异常。所以这个问题如何避免,就需要改进)
一、业务场景
首先我们的类对象在创建方法的时候如果使用static进行修饰,那么就可以实现可以在外部进行调用。
如下所示如果没有static修饰,外部将不能使用该方法:
public class GetToken {
public static String Signature(String access_token, String timestamp) {
//下面的"& timestamp"之间没有空格,由于文本转译原因这里需要加空格
String hashStr = "access_token=" + access_token + "& timestamp=" + timestamp;
//进行SHA1加密后生成signature
return SHA1(hashStr);
}
}
@Test
void test6() {
System.out.println("测试static修饰函数的");
String str = GetToken.Signature("1234", "2345");
System.out.println(str);
}
二.理论原理:
spring框架应用中有些静态方法需要依赖被容器管理的类,就像这样:
@Component
public class Test {
@Autowired
private static UserService userService;
public static void test() {
userService.test();
}
}
这样一定会报java.lang.NullPointerException: null异常。
二、原理剖析
静态变量、类变量不是对象的属性,而是一个类的属性,所以静态方法是属于类(class)的,普通方法才是属于实体对象(也就是New出来的对象)的,spring注入是在容器中实例化对象,所以不能使用静态方法。
而使用静态变量、类变量扩大了静态方法的使用范围。静态方法在spring是不推荐使用的,依赖注入的主要目的,是让容器去产生一个对象的实例,然后在整个生命周期中使用他们,同时也让testing工作更加容易。
一旦你使用静态方法,就不再需要去产生这个类的实例,这会让testing变得更加困难,同时你也不能为一个给定的类,依靠注入方式去产生多个具有不同的依赖环境的实例,这种static field是隐含共享的,并且是一种global全局状态,spring同样不推荐这样去做。
在Springframework里,我们是不能@Autowired一个静态变量,使之成为一个Spring bean的。为什么?其实很简单,因为当类加载器加载静态变量时,Spring上下文尚未加载。所以类加载器不会在bean中正确注入静态类,并且会失败。
三、解决方法
1、将@Autowire加到构造方法上
@Component
public class Test {
private static UserService userService;
@Autowired
public Test(UserService userService) {
Test.userService = userService;
}
public static void test() {
userService.test();
}
}
2、用@PostConstruct注解
@Component
public class Test {
private static UserService userService;
@Autowired
private UserService userService2;
@PostConstruct
public void beforeInit() {
userService = userService2;
}
public static void test() {
userService.test();
}
}