@Autowired和static的关系
一、发生的场景
好几次有个同事因为把static用到Spring的@Autowired上,导致注入的对象一直报空指针,他一直找不到错误在哪里,来问我,其实我以前也不知道这个问题,但我根据Spring容器的特点判定,他调用的对象与注入的对象不是一个对象,就告诉他:static的加载顺序是在@Autowired之前;之后查资料才知道其实不是这样。。。
二、原理剖析
静态变量、类变量不是对象的属性,而是一个类的属性,所以静态方法是属于类(class)的,普通方法才是属于实体对象(也就是New出来的对象)的,spring注入是在容器中实例化对象,所以不能使用静态方法。
而使用静态变量、类变量扩大了静态方法的使用范围。静态方法在spring是不推荐使用的,依赖注入的主要目的,是让容器去产生一个对象的实例,然后在整个生命周期中使用他们,同时也让testing工作更加容易。
一旦你使用静态方法,就不再需要去产生这个类的实例,这会让testing变得更加困难,同时你也不能为一个给定的类,依靠注入方式去产生多个具有不同的依赖环境的实例,这种static field是隐含共享的,并且是一种global全局状态,Spring同样不推荐这样去做。
三、解决方案
1、将@Autowire加到构造方法上
1
2
3
4
5
6
7
8
9
10
11
|
@Component public class Test { private static UserService userService; @Autowired public Test(UserService userService) { Test.userService = userService; } public static void test() { userService.test(); } } |
2、用@PostConstruct注解
1
2
3
4
5
6
7
8
9
10
11
12
13
|
@Component public class Test { private static UserService userService; @Autowired private UserService userService2; @PostConstruct public void beforeInit() { userService = userService2; } public static void test() { userService.test(); } } |
static方法使用@Autowired
set注入失败 构造器注入成功
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
@Component @Slf4j public class UserCookieInfoUtil { private static RedisTemplate<String, String> redisTemplate; private static JWTUtils jwtUtils; @Autowired public UserCookieInfoUtil(JWTUtils jwtUtils, RedisTemplate<String, String> redisTemplate) { UserCookieInfoUtil.jwtUtils = jwtUtils; UserCookieInfoUtil.redisTemplate = redisTemplate; } public static Map<String, Object> getCookieInfo(String token) { Map<String, Object> map = new HashMap<>(); try { String loginName = jwtUtils.getLoginName(token); String info = redisTemplate.opsForValue().get(Constants.TOKEN_USER_INFO + ":" + loginName); map = JSONObject.parseObject(info == null ? "" : info, Map. class ); } catch (Exception e){ log.error( "获取缓存中的登录信息失败:{}" , e); } return map; } } |
以上为个人经验,希望能给大家一个参考,也希望大家多多支持服务器之家。
原文链接:https://www.cnblogs.com/pangyangqi/p/10010866.html