Atitit SpringCache緩存使用
Atitit SpringCache緩存使用 艾提拉 attilax總結(jié)
?
1. Spring的抽象已經(jīng)做得夠好了,適合于大多數(shù)場景,非常復(fù)雜的就需要自己AOP實現(xiàn)了。 1
1.1. 設(shè)置配置文件支持 ?applicataion.xml 1
1.2. -------------通過注解去使用到Cache--- 3
1.3. @Cacheable-------使用這個注解的方法在執(zhí)行后會緩存其返回結(jié)果。 3
1.4. @CachePut?應(yīng)用到寫數(shù)據(jù)的方法上,如新增/修改方法,調(diào)用方法時會自動把相應(yīng)的數(shù)據(jù)放入緩存:? 4
1.4.1. @CacheEvict?即應(yīng)用到移除數(shù)據(jù)的方法上,如刪除方法,調(diào)用方法時會從緩存中移除相應(yīng)的數(shù)據(jù): 4
1.4.2. @Caching?組合多個Cache注解使用 4
1.4.3. 自定義緩存注解 4
1.5. 測試 4
2. 總結(jié) 5
2.1. key的設(shè)置 默認(rèn)以參數(shù)作為key 5
2.2. 定時刷新的問題 5
2.3. 不同的方法使用不同的緩存以及緩存時間 6
3. Spring cache獨立使用的模式 6
4. 不足之處 8
5. 參考資料 8
?
設(shè)置配置文件支持 ?applicataion.xml
1.?Spring的抽象已經(jīng)做得夠好了,適合于大多數(shù)場景,非常復(fù)雜的就需要自己AOP實現(xiàn)了。 1.1.?設(shè)置配置文件支持 ?applicataion.xml
?
<beans ??default-lazy-init="true"????xmlns="http://www.springframework.org/schema/beans"
???????xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
???????xmlns:aop="http://www.springframework.org/schema/aop"
???????xmlns:tx="http://www.springframework.org/schema/tx"
???????xmlns:context="http://www.springframework.org/schema/context"
??????xmlns:cache="http://www.springframework.org/schema/cache"
????????
?????????xsi:schemaLocation="http://www.springframework.org/schema/beans ???????????????????????http://www.springframework.org/schema/beans/spring-beans-3.1.xsd
???????????????????????http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd
???????????????????????http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd
????????????????????????http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd
?????????????????????????http://www.springframework.org/schema/cache
??????????????http://www.springframework.org/schema/cache/spring-cache.xsd" ????>
???
<bean id="cacheManager"
?
?????????????????class="org.springframework.cache.concurrent.ConcurrentMapCacheManager"?/>
?
?
?
?
1.2.?-------------通過注解去使用到Cache--- 1.3.?@Cacheable-------使用這個注解的方法在執(zhí)行后會緩存其返回結(jié)果。
應(yīng)用到讀取數(shù)據(jù)的方法上,即可緩存的方法,如查找方法:先從緩存中讀取,如果沒有再調(diào)用方法獲取數(shù)據(jù),然后把數(shù)據(jù)添加到緩存中:
?
package com.cnhis.cloudhealth.clinical.clidoctor.acolsetting.dao;
?
import java.net.URLDecoder;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentMap;
?
import org.springframework.cache.annotation.Cacheable;
import org.springframework.stereotype.Repository;
?
import com.alibaba.fastjson.JSON;
import com.cnhis.cloudhealth.clinical.util.SpringUtilV3_prjcli;
import com.cnhis.cloudhealth.commons.Mappers.ModelVo;
import com.cnhis.cloudhealth.commons.dao.BaseDao;
import com.google.common.collect.Maps;
@SuppressWarnings("rawtypes")
@Repository
public class IcolSettingDao extends BaseDao {
@Cacheable(value?=?{?"Cachename1"?})
public?Object?getSetting(Map?map)?{
try?{
return????getSqlSession().selectList("IcolSetting.select_datadic",?map);
}?catch?(Exception?e)?{
throw?new?RuntimeException(e);
}
//return new ModelVo();// Maps.newConcurrentMap();
}
?
1.4.?@CachePut?應(yīng)用到寫數(shù)據(jù)的方法上,如新增/修改方法,調(diào)用方法時會自動把相應(yīng)的數(shù)據(jù)放入緩存:? 1.4.1.?@CacheEvict?即應(yīng)用到移除數(shù)據(jù)的方法上,如刪除方法,調(diào)用方法時會從緩存中移除相應(yīng)的數(shù)據(jù):
?
1.4.2.?@Caching?組合多個Cache注解使用
有時候我們可能組合多個Cache注解使用;比如用戶新增成功后,我們要添加id-->user;username--->user;email--->user的緩存;此時就需要@Caching組合多個注解標(biāo)簽了。
?
如用戶新增成功后,添加id-->user;username--->user;email--->user到緩存;?
Java代碼??
1.?@Caching(??
2.?????????put?=?{??
3.?????????????????@CachePut(value?=?"user",?key?=?"#user.id"),??
4.?????????????????@CachePut(value?=?"user",?key?=?"#user.username"),??
5.?????????????????@CachePut(value?=?"user",?key?=?"#user.email")??
6.?????????}??
7.?)??
8.?public?User?save(User?user)?{??
1.4.3.?自定義緩存注解
比如之前的那個@Caching組合,會讓方法上的注解顯得整個代碼比較亂,此時可以使用自定義注解把這些注解組合到一個注解中,如:
?
?
1.5.?測試
SpringUtilV3_prjcli.cfgFileDir="C:\0wkspc\clis413\clinical\springtest_cli";
SpringUtilV3_prjcli.setLocations(cfgFileDir,"applicationContext-datasource.xml,onehis-dubbo.xml");
IcolSettingDao?d=(IcolSettingDao)?SpringUtilV3_prjcli.getBean(?IcolSettingDao.class);
//gene_barcode_test();
ConcurrentMap
Object?setting?=?d.getSetting(newConcurrentMap);
System.out.println(setting);
Object?setting2?=?d.getSetting(newConcurrentMap);
System.out.println(setting2);
?
?
2.?總結(jié) 2.1.?key的設(shè)置 默認(rèn)以參數(shù)作為key
?????? @Cacheable(value="緩存空間的名稱,xml中配置的" , key = "#spittle.id")--spittle是參數(shù)里面的spittle,如果不設(shè)置,就以參數(shù)作為key
??????? Spittle save(Spittle spittle);
2.2.?定時刷新的問題
·?緩存失效時間支持在方法的注解上指定
Spring Cache默認(rèn)是不支持在@Cacheable上添加過期時間的,可以在配置緩存容器時統(tǒng)一指定:
@Bean
ConcurrentMapCacheManager可以作為一種緩存方案,但不能設(shè)置過期,最大緩存條目等,需進(jìn)行改造。
???class="com.cnhis.cloudhealth.clinical.util.cache.MyConcurrentMapCacheManager">
????????
?
其實也可以timer to refresh
?
2.3.?不同的方法使用不同的緩存以及緩存時間
?
還可以使用不同的多個 CacheManager?每個,cm時間不同即可。。注意cache那么也不要重復(fù)了,可以增加個cm前綴命名空間。。
?
另外還提供了CompositeCacheManager用于組合CacheManager,即可以從多個CacheManager中輪詢得到相應(yīng)的Cache,如
Java代碼??
1.?
2.?????
3.?????????
4.?????????????
5.?????????????
6.???????????
7.???????
8.?????
9.???
當(dāng)我們調(diào)用cacheManager.getCache(cacheName) 時,會先從第一個cacheManager中查找有沒有cacheName的cache,如果沒有接著查找第二個,如果最后找不到,因為fallbackToNoOpCache=true,那么將返回一個NOP的Cache否則返回null。
?
Spring Cache抽象詳解,一篇很好的spring Cache的解釋文章,結(jié)合源碼更加讓人容易懂 - CSDN博客.html
?
不同的緩存可以指定cachename
不同的時間,則只好使用不同的定時器,清理指定的緩存
CacheManager?CacheManager1=??(CacheManager)?SpringUtilV3_prjcli.getBean(?CacheManager.class);
Cache?Cache1=CacheManager1.getCache("Cachename1");
Cache1.clear();
?
?
?
3.?Spring cache獨立使用的模式
?
package com.cnhis.cloudhealth.clinical.util.cache;
?
/*
?* Copyright 2002-2014 the original author or authors.
?*
?* Licensed under the Apache License, Version 2.0 (the "License");
?* you may not use this file except in compliance with the License.
?* You may obtain a copy of the License at
?*
?* ?????http://www.apache.org/licenses/LICENSE-2.0
?*
?* Unless required by applicable law or agreed to in writing, software
?* distributed under the License is distributed on an "AS IS" BASIS,
?* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
?* See the License for the specific language governing permissions and
?* limitations under the License.
?*/
?
import java.util.Collection;
import java.util.Collections;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.TimeUnit;
?
import org.springframework.cache.Cache;
import org.springframework.cache.CacheManager;
import org.springframework.cache.concurrent.ConcurrentMapCache;
?
import com.google.common.cache.CacheBuilder;
?
/**
?* {@link CacheManager} implementation that lazily builds {@link ConcurrentMapCache}
?* instances for each {@link #getCache} request. Also supports a ‘static‘ mode where
?* the set of cache names is pre-defined through {@link #setCacheNames}, with no
?* dynamic creation of further cache regions at runtime.
?*
?*
Note: This is by no means a sophisticated CacheManager; it comes with no
?* cache configuration options. However, it may be useful for testing or simple
?* caching scenarios. For advanced local caching needs, consider
?* {@link org.springframework.cache.guava.GuavaCacheManager} or
?* {@link org.springframework.cache.ehcache.EhCacheCacheManager}.
?*
?* @author Juergen Hoeller
?* @since 3.1
?* @see ConcurrentMapCache
?*/
public class MyConcurrentMapCacheManager implements CacheManager {
public static void main(String[] args) {
MyConcurrentMapCacheManager CacheManager2=new MyConcurrentMapCacheManager(20, 1000);
????CacheManager2.createConcurrentMapCache("cachename1");
????
????Cache Cache1=CacheManager2.getCache("cachename1");
????
????Cache1.put("k", "vv"); ???
????System.out.println(Cache1.get("k").get());
????System.out.println(Cache1.get("k2").get()); ??//if no key ?ret ?NullPointerException
????
}
?
4.?不足之處
?
當(dāng)然Spring Cache注解對于大多數(shù)場景夠用了,如果場景復(fù)雜還是考慮使用AOP吧;如果自己實現(xiàn)請考慮使用Spring Cache API進(jìn)行緩存抽象。
5.?參考資料
SpringCache緩存初探 - 沐魘 - 博客園.html
Spring Cache抽象詳解,一篇很好的spring Cache的解釋文章,結(jié)合源碼更加讓人容易懂 - CSDN博客.html (注解全局配置)