springboot中redis的緩存穿透問(wèn)題實(shí)現(xiàn)
什么是緩存穿透問(wèn)題??
我們使用redis是為了減少數(shù)據(jù)庫(kù)的壓力,讓盡量多的請(qǐng)求去承壓能力比較大的redis,而不是數(shù)據(jù)庫(kù)。但是高并發(fā)條件下,可能會(huì)在redis還沒(méi)有緩存的時(shí)候,大量的請(qǐng)求同時(shí)進(jìn)入,導(dǎo)致一大批的請(qǐng)求直奔數(shù)據(jù)庫(kù),而不會(huì)經(jīng)過(guò)redis。使用代碼模擬緩存穿透問(wèn)題如下:
首先是service里面的代碼:
@Service
public class NewsService {
@Autowired
private NewsDAO newsDAO;
//springboot自動(dòng)初始化,不需要我們進(jìn)行配置,直接注入到代碼中使用
@Autowired
private RedisTemplate<Object,Object> redisTemplate;
public /*synchronized*/ List<News> getLatestNews(int userId,int offset,int limit){
//設(shè)置序列化方式,防止亂碼
redisTemplate.setKeySerializer(new StringRedisSerializer());
//第一步:查詢(xún)緩存
News news= (News) redisTemplate.opsForValue().get("newsKey");
//判斷是否存在緩存
if(null == news){//查詢(xún)數(shù)據(jù)庫(kù)
news = newsDAO.selectByUserIdAndOffset(userId,offset,limit).get(0);
//
redisTemplate.opsForValue().set("newsKey",news);
System.out.println("進(jìn)入數(shù)據(jù)庫(kù)。。。。。。。。");
}else{
System.out.println("進(jìn)入緩存。。。。。。。。。");
}
return newsDAO.selectByUserIdAndOffset(userId,offset,limit);
}
}
然后是使用線程池在Controller里面對(duì)請(qǐng)求進(jìn)行模擬:
@Controller
public class HomeController {
@Autowired
UserService userService;
@Autowired
NewsService newsService;
//遇到的坑,如果不加method,頁(yè)面啟動(dòng)不起來(lái)。
@RequestMapping(value = "/home",method = {RequestMethod.GET, RequestMethod.POST})
@ResponseBody
public String index(Model model){
//這邊是可以讀出數(shù)據(jù)來(lái)的
//線程池------緩存穿透問(wèn)題的復(fù)現(xiàn)
ExecutorService executorService = Executors.newFixedThreadPool(8*2);
for(int i = 0;i < 50000;i++){
executorService.submit(new Runnable() {
@Override
public void run() {
List<News> newsList = newsService.getLatestNews(0,0,10);
}
});
}
List<News> newsList = newsService.getLatestNews(0,0,10);
News news=newsList.get(0);
return news.getImage();
}
}
結(jié)果如圖:大量的請(qǐng)求進(jìn)入數(shù)據(jù)庫(kù),那么如何解決這個(gè)問(wèn)題?

方法一、在方法上加鎖:
@Service
public class NewsService {
@Autowired
private NewsDAO newsDAO;
//springboot自動(dòng)初始化,不需要我們進(jìn)行配置,直接注入到代碼中使用
@Autowired
private RedisTemplate<Object,Object> redisTemplate;
//第一種方式:方法加鎖
public synchronized List<News> getLatestNews(int userId,int offset,int limit){
//設(shè)置序列化方式,防止亂碼
redisTemplate.setKeySerializer(new StringRedisSerializer());
//第一步:查詢(xún)緩存
News news= (News) redisTemplate.opsForValue().get("newsKey");
//判斷是否存在緩存
if(null == news){
//查詢(xún)數(shù)據(jù)庫(kù)
news = newsDAO.selectByUserIdAndOffset(userId,offset,limit).get(0);
//
redisTemplate.opsForValue().set("newsKey",news);
System.out.println("進(jìn)入數(shù)據(jù)庫(kù)。。。。。。。。");
}else{
System.out.println("進(jìn)入緩存。。。。。。。。。");
}
return newsDAO.selectByUserIdAndOffset(userId,offset,limit);
}
}
直接在方法上加鎖,保證每次只有一個(gè)請(qǐng)求可以進(jìn)入。但是這個(gè)方法存在一個(gè)缺陷,每次只有一個(gè)請(qǐng)求可以進(jìn)入,請(qǐng)求處理的速度變得相當(dāng)?shù)穆?,不利于系統(tǒng)的實(shí)時(shí)性。
方法二、使用雙重校驗(yàn)鎖:
@Service
public class NewsService {
@Autowired
private NewsDAO newsDAO;
//springboot自動(dòng)初始化,不需要我們進(jìn)行配置,直接注入到代碼中使用
@Autowired
private RedisTemplate<Object,Object> redisTemplate;
//第一種方式:方法加鎖
public /*synchronized*/ List<News> getLatestNews(int userId,int offset,int limit){
//設(shè)置序列化方式,防止亂碼
redisTemplate.setKeySerializer(new StringRedisSerializer());
//第一步:查詢(xún)緩存
News news= (News) redisTemplate.opsForValue().get("newsKey");
//判斷是否存在緩存
if(null == news){
//第二種方式:雙重檢測(cè)鎖
synchronized (this){
//查詢(xún)數(shù)據(jù)庫(kù)
news = newsDAO.selectByUserIdAndOffset(userId,offset,limit).get(0);
//
redisTemplate.opsForValue().set("newsKey",news);
System.out.println("進(jìn)入數(shù)據(jù)庫(kù)。。。。。。。。");
}
}else{
System.out.println("進(jìn)入緩存。。。。。。。。。");
}
return newsDAO.selectByUserIdAndOffset(userId,offset,limit);
}
}
這個(gè)方法比較好,雖然不能保證只有一個(gè)請(qǐng)求請(qǐng)求數(shù)據(jù)庫(kù),但是當(dāng)?shù)谝慌?qǐng)求進(jìn)來(lái),第二批之后的所有請(qǐng)求全部會(huì)在緩存取數(shù)據(jù)。
到此這篇關(guān)于springboot中redis的緩存穿透問(wèn)題實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)springboot redis緩存穿透內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java 動(dòng)態(tài)編譯在項(xiàng)目中的實(shí)踐分享
在 Java 中,動(dòng)態(tài)編譯是指在運(yùn)行時(shí)動(dòng)態(tài)地編譯 Java 源代碼,生成字節(jié)碼,并加載到 JVM 中執(zhí)行,動(dòng)態(tài)編譯可以用于實(shí)現(xiàn)動(dòng)態(tài)代碼生成、動(dòng)態(tài)加載、插件化等功能,本文將給大家分享一下Java 動(dòng)態(tài)編譯在項(xiàng)目中的實(shí)踐,感興趣的同學(xué)跟著小編一起來(lái)看看吧2023-07-07
解決Druid動(dòng)態(tài)數(shù)據(jù)源配置重復(fù)刷錯(cuò)誤日志的問(wèn)題
使用druid數(shù)據(jù)庫(kù)連接池實(shí)現(xiàn)動(dòng)態(tài)的配置數(shù)據(jù)源功能,在配置過(guò)程中出現(xiàn)一個(gè)問(wèn)題既然是用戶(hù)自己配置的數(shù)據(jù)源,就無(wú)法避免輸入錯(cuò)誤,連接失敗等情況,關(guān)于這個(gè)問(wèn)題怎么處理呢,今天小編通過(guò)本文給大家詳細(xì)說(shuō)明下,感興趣的朋友一起看看吧2021-05-05
FluentMybatis實(shí)現(xiàn)mybatis動(dòng)態(tài)sql拼裝和fluent api語(yǔ)法
本文主要介紹了FluentMybatis實(shí)現(xiàn)mybatis動(dòng)態(tài)sql拼裝和fluent api語(yǔ)法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-08-08
Java String.replace()方法"無(wú)效"的原因及解決方式
這篇文章主要介紹了Java String.replace()方法"無(wú)效"的原因及解決方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-08-08
Springboot中@Async異步,實(shí)現(xiàn)異步結(jié)果合并統(tǒng)一返回方式
這篇文章主要介紹了Springboot中@Async異步,實(shí)現(xiàn)異步結(jié)果合并統(tǒng)一返回方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-09-09
java實(shí)現(xiàn)表格數(shù)據(jù)的存儲(chǔ)
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)表格數(shù)據(jù)的存儲(chǔ),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-04-04
詳解通過(guò)maven運(yùn)行項(xiàng)目的兩種方式
這篇文章主要介紹了通過(guò)maven運(yùn)行項(xiàng)目的兩種方式,給大家提到了通過(guò)tomcat的方式來(lái)啟動(dòng)maven項(xiàng)目的方法,通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),需要的朋友可以參考下2021-12-12
JAVA Stack詳細(xì)介紹和示例學(xué)習(xí)
JAVA Stack是棧。它的特性是:先進(jìn)后出(FILO, First In Last Out)。2013-11-11
布隆過(guò)濾器的原理以及java 簡(jiǎn)單實(shí)現(xiàn)
這篇文章主要介紹了布隆過(guò)濾器的原理以及java 簡(jiǎn)單實(shí)現(xiàn),幫助大家更好的理解和學(xué)習(xí)Java,感興趣的朋友可以了解下2020-11-11

