基于hibernate二级缓存的论坛缓存方案

[b]前言[/b]
hibernate查询缓存很难用,更新某个实体后会导致与此实体相关的所有查询缓存失效,很不爽;
此方案建立在hibernate二级缓存的基础上
[b]目的[/b]
缓存帖子的每一页(即回帖)
[b]何时写缓存?何时清理缓存?[/b]
读取帖子第N页是缓存之,有回帖 修改回帖 删除回帖时清理缓存
[b]缓存结构[/b]
/**
* 帖子分页缓存。
* cache key为帖子id;
* cache value为Map,key为页号; value为post id串,如‘2#3#4#5’
*/
private static Map<Integer, Map<Integer, String>> threadPostCache = new LRULinkedHashMap<Integer, Map<Integer, String>>(10000);


外层map以帖子id为主键,内层map以页号为主键,内层map的value为回帖id串 如‘2#3#4#5’
外层Map采用LRULinkedHashMap

真正的回帖数据在hibernate二级缓存中,下面是读取二级缓存的逻辑代码

	@Override
public Page<ThreadPost> getPage(int pageNo, int pageSize, Integer postNum, Integer threadId) {
pageNo = pageNo == 0 ? 1 : pageNo;
Page<ThreadPost> page = null;

Map<Integer, String> pageMap = threadPostCache.get(threadId);
if(pageMap == null){
pageMap = new HashMap<Integer, String>();
}

String postIdString = pageMap.get(pageNo);
if(postIdString == null){
//从数据库中加载Page
page = threadPostManager.getPage(pageNo, pageSize, postNum, threadId);
//生成postIdString
postIdString = joinPostId(page.getDataSet());
pageMap.put(pageNo, postIdString);
threadPostCache.put(threadId, pageMap);
}else{
page = buildPage(pageNo, pageSize, postNum, postIdString);
}
return page;
}


//构造一个Page对象
private Page<ThreadPost> buildPage(int pageNo, int pageSize, Integer postNum, String postIdString){
if(StringUtils.isBlank(postIdString)){
return Page.EMPTY_PAGE;
}

List<ThreadPost> list = new ArrayList<ThreadPost>(postIdString.length());
for(String postId : postIdString.split("#")){
//[color=red]从hibernate二级缓存中读取[/color]
list.add(threadPostManager.get(Integer.valueOf(postId)));
}

return new Page<ThreadPost>(pageNo, postNum, pageSize, list);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值