新聞中心
環(huán)境:springboot2.3.12.RELEASE + JSR107 + Ehcache + JPA

成都創(chuàng)新互聯(lián)專(zhuān)注為客戶(hù)提供全方位的互聯(lián)網(wǎng)綜合服務(wù),包含不限于網(wǎng)站設(shè)計(jì)、成都網(wǎng)站建設(shè)、通榆網(wǎng)絡(luò)推廣、微信平臺(tái)小程序開(kāi)發(fā)、通榆網(wǎng)絡(luò)營(yíng)銷(xiāo)、通榆企業(yè)策劃、通榆品牌公關(guān)、搜索引擎seo、人物專(zhuān)訪(fǎng)、企業(yè)宣傳片、企業(yè)代運(yùn)營(yíng)等,從售前售中售后,我們都將竭誠(chéng)為您服務(wù),您的肯定,是我們最大的嘉獎(jiǎng);成都創(chuàng)新互聯(lián)為所有大學(xué)生創(chuàng)業(yè)者提供通榆建站搭建服務(wù),24小時(shí)服務(wù)熱線(xiàn):028-86922220,官方網(wǎng)址:www.cdcxhl.com
Spring 框架從 3.1 開(kāi)始,對(duì) Spring 應(yīng)用程序提供了透明式添加緩存的支持。和事務(wù)支持一樣,抽象緩存允許一致地使用各種緩存解決方案,并對(duì)代碼的影響最小。從 Spring4.1 版本開(kāi)始,緩存抽象支持了 JSR-107 注釋和更多自定義選項(xiàng),從而得到了顯著的改進(jìn)。
方式1:直接使用spring的注解來(lái)實(shí)現(xiàn)緩存
spring提供了如下注解:
@Cacheable 觸發(fā)緩存機(jī)制
@CacheEvict 觸發(fā)緩存回收
@CachePut 更新緩存,而不會(huì)影響方法的執(zhí)行
@Caching 組合多個(gè)緩存操作到一個(gè)方法
@CacheConfig 類(lèi)級(jí)別共享系誒常見(jiàn)的緩存相關(guān)配置
org.springframework.boot
spring-boot-starter-cache
首先在Service對(duì)應(yīng)的方法是添加注解:
@Service
public class StorageService {
@Resource
private StorageRepository sr ;
@Cacheable(value = {"cache_storage"}, keyGenerator = "storageKey")
public Storage getStorage(Long id) {
return sr.findById(id).get() ;
}
}// 這里的keyGenerator是你自定義Key生成的Bean名稱(chēng)
@Component("storageKey")
public class StorageKeyGenerator implements KeyGenerator {
private static final String KEY_PREFIX = "storage_" ;
@Override
public Object generate(Object target, Method method, Object... params) {
StringBuilder sb = new StringBuilder() ;
for (Object param : params) {
sb.append(param) ;
}
return KEY_PREFIX + sb.toString() ;
}
}web接口:
@RestController
@RequestMapping("/storages")
public class StorageController {
@Resource
private StorageService storageService ;
@GetMapping("/{id}")
public Object get(@PathVariable("id") Long id) {
return storageService.getStorage(id) ;
}
}測(cè)試:
第一次訪(fǎng)問(wèn)接口,查看控制臺(tái)輸出了sql語(yǔ)句:
圖片
再次訪(fǎng)問(wèn)接口,發(fā)現(xiàn)控制臺(tái)沒(méi)有再輸出任何sql,說(shuō)明我們的緩存生效了(這里你也可以把這里的注解注釋了來(lái)看效果)。關(guān)于這里的更新緩存,刪除緩存就不演示了。接下來(lái)完整的演示下JSR107規(guī)范中的注解演示:
注意在這些注釋中我們是可以使用SpEL表達(dá)式的:
圖片
方式2:使用JSR107和Ehcache
先來(lái)看看Spring與JSR107注解的對(duì)照表:
圖片
pom.xml中加入依賴(lài):
org.springframework.boot
spring-boot-starter-cache
mysql
mysql-connector-java
org.ehcache
ehcache
javax.cache
cache-api
Service類(lèi):
@Service
public class StorageService {
@Resource
private StorageRepository sr ;
// 這里的 @CacheValue 說(shuō)明是要緩存的參數(shù)值。
@Transactional
@CachePut(cacheName = "cache_storage", cacheKeyGenerator = JCacheKeyGenerator.class)
public Storage save(@CacheValue Storage storage) {
return sr.saveAndFlush(storage) ;
}
@CacheResult(cacheName = "cache_storage", cacheKeyGenerator = JCacheKeyGenerator.class)
public Storage getStorage(Long id) {
return sr.findById(id).get() ;
}
@Transactional
@CacheRemove(cacheName = "cache_storage", cacheKeyGenerator = JCacheKeyGenerator.class)
public void removeStorage(Long id) {
sr.deleteById(id) ;
}
@Transactional
@CachePut(cacheName = "cache_storage", cacheKeyGenerator = JCacheKeyGenerator.class)
public Storage updateStorage(@CacheValue Storage storage) {
return sr.saveAndFlush(storage) ;
}
}
// 注意這里的cacheKeyGenerator 必須全部用同一個(gè),
// 跟蹤了下源碼是用的對(duì)應(yīng)的類(lèi)名key來(lái)查找對(duì)應(yīng)的緩存的;一開(kāi)始我沒(méi)有用同一個(gè)始終不正確。。
// 看下圖跟蹤的代碼:圖片
這里必須要一樣哦cacheKeyGenerator
緩存Key:JCacheKeyGenerator.java
public class JCacheKeyGenerator implements CacheKeyGenerator {
private static final String KEY_PREFIX = "storage_" ;
@Override
public GeneratedCacheKey generateCacheKey(
CacheKeyInvocationContext extends Annotation> cacheKeyInvocationContext) {
CacheInvocationParameter[] params = cacheKeyInvocationContext.getAllParameters() ;
StringBuilder sb = new StringBuilder() ;
for (CacheInvocationParameter param : params) {
if (param.getValue() instanceof Storage) {
Storage s = (Storage) param.getValue() ;
sb.append(s.getId()) ;
} else {
sb.append((Long)param.getValue()) ;
}
}
return new StorageGeneratedCacheKey(KEY_PREFIX + sb.toString()) ;
}
private static class StorageGeneratedCacheKey implements GeneratedCacheKey {
private static final long serialVersionUID = 1L;
private String key ;
public StorageGeneratedCacheKey(String key) {
this.key = key ;
}
@Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result + ((key == null) ? 0 : key.hashCode());
return result;
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
StorageGeneratedCacheKey other = (StorageGeneratedCacheKey) obj;
if (key == null) {
if (other.key != null)
return false;
} else if (!key.equals(other.key))
return false;
return true;
}
}
}application.yml配置:
spring:
cache:
cacheNames:
- cache_storage
ehcache:
config: classpath:ehcache.xmlehcache.xml
測(cè)試增刪改:
先添加個(gè)數(shù)據(jù):
圖片
圖片
成功添加ID為4的信息,Service中的save方法中我們添加了@CachePut注解,接下來(lái)我們查詢(xún)ID為4的信息,看看控制臺(tái)是否會(huì)生成SQL語(yǔ)句。
圖片
圖片
控制臺(tái)沒(méi)有增加任何的SQL語(yǔ)句,說(shuō)明save方法加的@CachePut生效了。
接著做刪除操作:
圖片
圖片
ID為4的刪除了,接下來(lái)再做查詢(xún)看看:
圖片
這說(shuō)明刪除了數(shù)據(jù)后,緩存也做了刪除。這里生成了查詢(xún)語(yǔ)句。
新聞名稱(chēng):SpringBoot中使用Cache提升接口性能詳解
鏈接分享:http://m.5511xx.com/article/dpcpgcd.html


咨詢(xún)
建站咨詢(xún)
