声明,如果你不关心java缓存解决方案的全貌,只是急着解决问题,请略过背景部分。
在互联网应用中,由于并发量比传统的企业级应用会高出很多,所以处理大并发的问题就显得尤为重要。在硬件资源一定的情况下,在软件层面上解决高并发问题会比较经济实惠一些。解决并发的根本在于提高系统的响应时间与单位时间的吞吐量。解决问题的思路可分两个维度,一是提高系统的单位时间内的运算效率(比如集群),二是减少系统不必要的开支(比如缓存)。缓存又会分为客户端缓存与服务器端缓存,本文就javaEE项目的服务器端缓存方案展开介绍。
根据网上资料Java缓存解决方案有多种,用的比较多,中文资料也比较多的有:oscache、ehcache。如何选择参考了网上前辈们的评价。如下:
http://blog.sina.com.cn/s/blog_46d5caa40100ka9z.html
由于我们系统采用springMVC,所以在选定ehcache后,就着重研究ehcache与SpringMVC的整合,注解当然是要用到的。参考资料:
http://hanqunfeng.iteye.com/blog/603719
在了解了spring与ehcache整合之后发现这东西是不是很简单,但是笔者在查找资料的时候又发现了更简单解决方案,google为spring与ehcache整合提供了封装包,参考资料如下:
http://luck332.iteye.com/blog/1001783
http://code.google.com/p/ehcache-spring-annotations/
http://blog.goyello.com/2010/07/29/quick-start-with-ehcache-annotations-for-spring/(推荐参考,虽然是e文的,但是基本可以看懂)
进入正题。
SpringMVC + ehcache(google ehcache-spring-annotations),基于注解解决服务器端数据缓存。
ehcache-2.7.1.jar
slf4j-api-1.6.6.jar
slf4j-jdk14-1.6.6.jar
down下来之后lib里面会有以上三个包(这两个slf4j的包一般项目里会有,换成最新的版本即可),下载地址如下:
ehcache-spring-annotations-1.2.0.jar
ehcache-spring-annotations-1.2.0-sources.jar
down下来之后,从文件夹取以上2个包,其他的包忽视。下载地址:
当然google也提供了开发版的包,里面提供了源码。jar还支持二维码下载,下载到手机上?不知道google咋想的。。。
给你需要缓存的方法加上如下这句
cacheName里面对应ehcache.xml中配置的<cache name="mallListCache" ,
这里我想说,网上很多人都是加到dao上的,我是直接加到service(业务层)里面的方法上,因为我的业务层方法会对数据做处理,而我需要缓存整个处理结果,所以加到service里面的方法上是可以的。
cacheName里面对应缓存里面的名称,removeAll=true 这句是必须加的,否则无法清除缓存。
1.对于清除缓存的方法,ehcache提供了两种,一种是在ehcache.xml中配置的时间过后自动清除,一种是在数据发生变化后触发清除。个人感觉第二种比较好。可以将
@TriggersRemove(cacheName="mallListCache",removeAll=true)
这句代码加到service里面的添加、删除、修改方法上。这样只要这几个方法有调用,缓存自动清除。
---------------------------------------------------
1
2
3
4
|
ehcache-
2.7
.
5
.jar(主程序)
ehcache-spring-annotations-
1.2
.
0
.jar(注解)
guava-r09.jar(依赖)
slf4j-api-
1.6
.
6
.jar(依赖)
|
头部
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
xmlns:cache=
"http://www.springframework.org/schema/cache"
xsi:schemaLocation
http:
//www.springframework.org/schema/cache
http:
//www.springframework.org/schema/cache/spring-cache-3.2.xsd
<!-- 缓存配置 -->
<!-- 启用缓存注解功能(请将其配置在Spring主配置文件中) -->
<cache:annotation-driven cache-manager=
"cacheManager"
/>
<!-- Spring自己的基于java.util.concurrent.ConcurrentHashMap实现的缓存管理器(该功能是从Spring3.
1
开始提供的) -->
<!-- <bean id=
"cacheManager"
class
=
"org.springframework.cache.support.SimpleCacheManager"
>
<property name=
"caches"
> <set> <bean name=
"myCache"
class
=
"org.springframework.cache.concurrent.ConcurrentMapCacheFactoryBean"
/>
</set> </property> </bean> -->
<!-- 若只想使用Spring自身提供的缓存器,则注释掉下面的两个关于Ehcache配置的bean,并启用上面的SimpleCacheManager即可 -->
<!-- Spring提供的基于的Ehcache实现的缓存管理器 -->
<bean id=
"cacheManagerFactory"
class
=
"org.springframework.cache.ehcache.EhCacheManagerFactoryBean"
>
<property name=
"configLocation"
value=
"classpath:ehcache.xml"
/>
</bean>
<bean id=
"cacheManager"
class
=
"org.springframework.cache.ehcache.EhCacheCacheManager"
>
<property name=
"cacheManager"
ref=
"cacheManagerFactory"
/>
</bean>
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
<ehcache>
<diskStore path=
"java.io.tmpdir"
/>
<defaultCache
maxElementsInMemory=
"1000"
eternal=
"false"
timeToIdleSeconds=
"120"
timeToLiveSeconds=
"120"
overflowToDisk=
"false"
/>
<cache name=
"myCache"
maxElementsOnDisk=
"20000"
maxElementsInMemory=
"2000"
eternal=
"false"
overflowToDisk=
"true"
diskPersistent=
"true"
/>
<cache name=
"cacheTest"
maxElementsOnDisk=
"20000"
maxElementsInMemory=
"2000"
eternal=
"false"
overflowToDisk=
"true"
diskPersistent=
"true"
/>
</ehcache>
|
cache一般用在和数据库交互的地方service
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
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
|
示例
package
com.service;
import
java.util.Date;
import
java.util.HashMap;
import
java.util.LinkedHashMap;
import
java.util.List;
import
java.util.Map;
import
javax.annotation.Resource;
import
org.apache.commons.logging.Log;
import
org.apache.commons.logging.LogFactory;
import
org.springframework.cache.annotation.CacheEvict;
import
org.springframework.cache.annotation.CachePut;
import
org.springframework.cache.annotation.Cacheable;
import
org.springframework.stereotype.Service;
import
com.aft.site.yey.dao.NoticeDao;
import
com.aft.site.yey.entity.Notice;
import
com.app.jdbc.core.ToolUtil;
/**
* @author: yangyang 2013年10月21日
* @since JDK 1.6
*/
@Service
(
"XXXNoticeService"
)
public
class
NoticeService {
private
static
Log logger = LogFactory.getLog(NoticeService.
class
);
@Resource
(name =
"XXXNoticeDao"
)
private
NoticeDao dao;
/**
* status = 0 指未删除
* */
@Cacheable
(value =
"cacheTest"
,key=
"'noticelist'"
)
public
List<Notice> topN(
int
begin,
int
end) {
LinkedHashMap<String, String> orderby =
new
LinkedHashMap<String, String>();
orderby.put(
"publish_time"
,
"desc"
);
Map<String, String> where =
new
HashMap<String, String>();
where.put(dao.STATUS +
" = ? "
, dao.NORMAL_CODE.toString());
//TODO:delete
System.out.println(
"list:"
);
logger.info(
"[list ]"
);
return
dao.find(where, orderby, begin, end);
}
@CacheEvict
(value =
"cacheTest"
,key=
"'noticelist'"
)
public
void
delete(String id) {
//TODO:delete
System.out.println(
"delete:"
);
logger.info(
"delete "
);
dao.delete(id,
false
);
}
@CacheEvict
(value =
"cacheTest"
, allEntries =
true
)
public
void
save(Notice notice) {
notice.setRowid(ToolUtil.getUUID());
notice.setStatus(dao.NORMAL_CODE);
notice.setPublish_time(
new
Date());
// TODO:yeyid的获得方式
notice.setYey_id(
"123"
);
dao.insert(notice);
//TODO:delete
System.out.println(
"save:"
);
logger.info(
"save "
);
}
public
Notice get(String id) {
return
dao.findById(id);
}
//@CachePut(value = "cacheTest",key="#notice.getRowid()")
public
void
update(Notice notice) {
Map<String, String> set =
new
HashMap<String, String>();
LinkedHashMap<String, String> where =
new
LinkedHashMap<String, String>();
set.put(
"title"
, notice.getTitle());
set.put(
"author"
, notice.getAuthor());
set.put(
"content"
, notice.getContent());
where.put(dao.getIdColumnName() +
"=?"
, notice.getRowid());
dao.update(set, where);
System.out.println(
"update:"
);
logger.info(
"update "
);
}
}
|
cache主要注解使用:@Cacheable,@CacheEvict,@CachePut
缓存是这样的,取值时在方法(A)调用前查一下缓存中是否有目标值,缓存存在的话直接从缓存中拿出不再去执行方法(A),这也是最基本的@Cacheable的概念;
缓存中有值需要更新怎么办?使用@CacheEvict来更新,这个注解的意思是删除掉缓存里面的某个值,从而达到更新缓存的效果。关于缓存更新,例如,取topN个对象,第一次取的时候比如是前1~10个,缓存中存这1~10的一个集合对象,第二次取的时候直接从缓存中拿,这没问题,现在是这样的,假设数据库中删除了1~10个元素中的任意一个值,这样数据库中的topN与缓存中的topN就不同步了,下次你在前台取topN的时候,因为缓存里面有这个对象,根据之前的介绍(取值时在方法(A)调用前查一下缓存中是否有目标值,缓存存在的话直接从缓存中拿出不再去执行方法(A)),方法A被略过,查的值不是真正的topN了,因此需要在add或者delete之后删除掉原来的缓存,保持数据一致。其他情景请自行考虑。
根据缓存的特性,如何做到既要保证方法被调用,又希望结果被缓存呢?直接使用@CachePut,他与@Cacheable的区别就在与方法是会被执行的。
注解里面属性解释,@Cacheable 与@CachePut一样, @CacheEvict还有和删除有关的两个属性:
value:缓存的名称,在spring配置文件中定义,必须指定至少一个
key:缓存的key,(缓存是键值对儿)可以为空,如果指定要按照 SpEL 表达式编写,如果不指定,则缺省按照方法的所有参数进行组合
condition:缓存的条件,可以为空,使用SpEL 编写,返回 true 或者 false,只有为 true 才进行缓存
allEntries:是否清空所有缓存内容,缺省为 false,如果指定为 true,则方法调用后将立即清空所有缓存
beforeInvocation:是否在方法执行前就清空,缺省为 false,如果指定为 true,则在*方法还没有执行的时候就清空缓存,缺省情况下为false,这样如果方法执行抛出异常,则不会清空缓存
集群的同步问题,未完待续。
http://www.ibm.com/developerworks/cn/opensource/os-cn-spring-cache/
http://docs.spring.io/spring/docs/3.1.0.M1/spring-framework-reference/html/cache.html
http://my.oschina.net/coolfire368/blog/123377
http://blog.csdn.net/jadyer/article/details/12257865
------------------------------------------------------------
由于这两天用的springmvc 和 mybatis 的搭建的web 框架 然后准备用缓存数据,就简单记录下
准备:
这个可以在https://code.google.com/p/ehcache-spring-annotations/ 下载,下载之后拿出来要用到的jar包
下载的压缩包中的注解包 ehcache-spring-annotations-1.2.0.jar 下载的压缩包中lib 目录下的 ehcache-core-2.4.5.jar guava-r09.jar
由于其他的一些包都在spring mvc 中已经有了,只需要添加这几个进去。
在src 目录下新增配置 ehcache.xml 如下:
<?xml version="1.0" encoding="UTF-8"?> <!-- /** * * 缓存配置 * @author yq * @date 2014.9.10 * */ --> <ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd" updateCheck="false"> <diskStore path="java.io.tmpdir" /> <defaultCache eternal="false" maxElementsInMemory="1000" overflowToDisk="false" diskPersistent="false" timeToIdleSeconds="0" timeToLiveSeconds="600" memoryStoreEvictionPolicy="LRU" /> <cache name="CustomerCache" eternal="false" maxElementsInMemory="100" overflowToDisk="false" diskPersistent="false" timeToIdleSeconds="0" timeToLiveSeconds="300" memoryStoreEvictionPolicy="LRU" /> </ehcache>
此配置中 可以有多个cache ,方便管理我们程序中的cache。
1 <ehcache:annotation-driven cache-manager="ehCacheManager" /> 2 <bean id="ehCacheManager" class="org.springframework.cache.ehcache.EhCacheManagerFactoryBean"> 3 <property name="configLocation" value="classpath:ehcache.xml" /> 4 </bean>
接下来就到我们的代码 Service 层 试用缓存了:
@Cacheable(cacheName = "CustomerCache") public Customer GetCustomerByCid(int cid){ System.out.print("----------------------------------------------------------"); return cd.GetCustomerByCid(cid); } //修改客户 @TriggersRemove(cacheName ={"CustomerCache"},removeAll=true) public int UpdateCustomer(Customer cus){ return cd.UpdateCustomer(cus); }
注意包的import ,是这两个
import com.googlecode.ehcache.annotations.Cacheable; import com.googlecode.ehcache.annotations.TriggersRemove;
以上就已经搭建好了。
当我们两次调用 service 中的 GetCustomerByCid 方法时候控制台只有一次sql打印表示成功了