spring中bean的构造函数,Autowired(Value)注入与@PostConstruct调用顺序
最近在项目开发中遇到这样一个需求,由于元数据在短时间内被客户端多次读取,因此希望直接将数据存储到内存,以减少网络开销,借助guava cache于是有了下面这个类
/**
* Created on 2018/10/18
*/
@Component
public class CacheUtil {
@Autowired
CaseGraphService caseGraphService;
@Value("${cache.expire.duration}")
long expireDuration;
private Cache<Long, CaseGraphDTO> metaNodeCache = CacheBuilder
.newBuilder()
.maximumSize(1000)
.expireAfterAccess(expireDuration, TimeUnit.MINUTES) //设置过期时间
.build();
public CaseGraphDTO getMetaNode(long caseId) throws ExecutionException {
return metaNodeCache.get(caseId, () -> caseGraphService.getCaseGraph(caseId));
}
public void removeMetaNodeByKey(long caseId){
metaNodeCache.invalidate(caseId);
}
public void removeMetaNodeAll(){
metaNodeCache.invalidateAll();
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
我们在另一个类中注入CacheUtil并调用它的getMetaNode方法,类似于这样:
@Component
public class TestComponent {
@Autowired
CacheUtil cacheUtil;
public CaseGraphDTO getMetaData(long caseId){
return cacheUtil.getMetaNode(caseId);
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
当我们第一次调用getMetaNode时,cache使用caseGraphService获取元数据,而后将这一元数据存放的cache中,我们在CacheUtil的getMetaNode方法中添加两行代码来测试一下:
public CaseGraphDTO getMetaNode(long caseId) throws ExecutionException {
//metaNodeCache的get方法,如果缓存中已有数据,直接返回数据,如果没有则通过Callable方法获取存入缓存再返回数据
CaseGraphDTO caseGraphDTO = metaNodeCache.get(caseId, () -> caseGraphService.getCaseGraph
(caseId));
//getIfPresent方法,如果存在数据即返回
CaseGraphDTO caseGraphDTO1 = metaNodeCache.getIfPresent(caseId);
System.out.println(caseGraphDTO1);
return caseGraphDTO;
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
启动application, 打开断点调试:
可以看到caseGraphDTO1为null,而不是我们预想的元数据;metaNodeCache中localCache的大小为0,也就是根本没有缓存任何数据。
而后我采用另一种方式@PostConstruct的方式来初始化metaNodeCache, 代码如下:
@Component
public class CacheUtil {
@Autowired
CaseGraphService caseGraphService;
@Value("${cache.expire.duration}")
long expireDuration;
private Cache<Long, CaseGraphDTO> metaNodeCache;
@PostConstruct
private void init(){
metaNodeCache = CacheBuilder
.newBuilder()
.maximumSize(1000)
.expireAfterAccess(expireDuration, TimeUnit.MINUTES)
.build();
}
public CaseGraphDTO getMetaNode(long caseId) throws ExecutionException {
//metaNodeCache的get方法,如果缓存中已有数据,直接返回数据,如果没有则通过Callable方法获取存入缓存再返回数据
CaseGraphDTO caseGraphDTO = metaNodeCache.get(caseId, () -> caseGraphService.getCaseGraph
(caseId));
//getIfPresent方法,如果存在数据即返回
CaseGraphDTO caseGraphDTO1 = metaNodeCache.getIfPresent(caseId);
System.out.println(caseGraphDTO1);
return caseGraphDTO;
}
public void removeMetaNodeByKey(long caseId){
metaNodeCache.invalidate(caseId);
}
public void removeMetaNodeAll(){
metaNodeCache.invalidateAll();
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
再进行调试:
这时我们看到caseGraphDTO1中有了数据,并且metaNodeCache中localCache的大小也变成了1.
那么直接初始化成员变量和使用postConstruct来初始化二者的区别是什么呢?
要将对象p注入到对象a,那么首先就必须得生成对象p与对象a,才能执行注入。所以,如果一个类A中有个成员变量p被@Autowired注解,那么@Autowired注入是发生在A的构造方法执行完之后的。
如果想在生成对象时候完成某些初始化操作,而偏偏这些初始化操作又依赖于依赖注入,那么就无法在构造函数中实现。为此,可以使用@PostConstruct注解一个方法来完成初始化,@PostConstruct注解的方法将会在依赖注入完成后被自动调用。
事情似乎明朗起来了,虽然我们在初始化metaNodeCache时没有使用autowired进来的caseGraphService,但我们使用了@Value来注入缓存过期时间(配置中这个值为60)。让我们再来断点调试一下,当使用直接初始化成员变量的方式时,@Value("${cache.expire.duration}")注入的过期时间是多少,如下:
我们看到expireDuration的值为0,而不是配置中的60,对于cache的含义即为缓存中的数据被读取后0分钟使其失效,等同于立即失效。所以导致上文我们的缓存中始终没有数据。
让我们再来看看使用PostConstruct初始化时,这个过期时间的值:
此时看到expireDuration的值为60.
因此我们需要记住,构造函数,Autowired(Value),PostConstruct的执行顺序为:
Constructor >> Autowired >> PostConstruct
如果初始化成员变量需要使用注入进来的对象或者值,那么应该放在被PostConstruct注解的方法中去做
原文地址:https://www.cnblogs.com/jpfss/p/11533597.html
- java教程
- Java快速入门
- Java 开发环境配置
- Java基本语法
- Java 对象和类
- Java 基本数据类型
- Java 变量类型
- Java 修饰符
- Java 运算符
- Java 循环结构
- Java 分支结构
- Java Number类
- Java Character类
- Java String类
- Java StringBuffer和StringBuilder类
- Java 数组
- Java 日期时间
- Java 正则表达式
- Java 方法
- Java 流(Stream)、文件(File)和IO
- Java 异常处理
- Java 继承
- Java 重写(Override)与重载(Overload)
- Java 多态
- Java 抽象类
- Java 封装
- Java 接口
- Java 包(package)
- Java 数据结构
- Java 集合框架
- Java 泛型
- Java 序列化
- Java 网络编程
- Java 发送邮件
- Java 多线程编程
- Java Applet基础
- Java 文档注释
- 如何通过Cloudera Manager页面自定义图表
- 【-Flutter组件篇- 】1.20新增组件InteractiveViewer
- Android使用SoundPool实现播放音效
- 【- FlutterUnit重大更新 -】Flutter要点集录.md
- Android实现美团外卖底部导航栏动画
- Kudu遇到的问题
- 【 -Flutter自定义组件- 】Wrapper组件,包裹装饰你的一切
- Android Shape属性创建环形进度条
- Ranger同步ldap组问题
- Android系统添加自定义鼠标样式通过按键切换实例详解
- Impala MetaData问题
- 聚焦 Android 11: UI 与 Compose
- 如何给Flutter界面切换实现点特效
- 恢复 RecyclerView 的滚动位置
- android实现指纹识别功能