亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術文章
文章詳情頁

springboot中redis的緩存穿透問題實現

瀏覽:2日期:2023-03-25 17:00:02
什么是緩存穿透問題??

我們使用redis是為了減少數據庫的壓力,讓盡量多的請求去承壓能力比較大的redis,而不是數據庫。但是高并發條件下,可能會在redis還沒有緩存的時候,大量的請求同時進入,導致一大批的請求直奔數據庫,而不會經過redis。使用代碼模擬緩存穿透問題如下:

首先是service里面的代碼:

@Servicepublic class NewsService { @Autowired private NewsDAO newsDAO; //springboot自動初始化,不需要我們進行配置,直接注入到代碼中使用 @Autowired private RedisTemplate<Object,Object> redisTemplate; public /*synchronized*/ List<News> getLatestNews(int userId,int offset,int limit){ //設置序列化方式,防止亂碼 redisTemplate.setKeySerializer(new StringRedisSerializer()); //第一步:查詢緩存 News news= (News) redisTemplate.opsForValue().get('newsKey'); //判斷是否存在緩存 if(null == news){//查詢數據庫news = newsDAO.selectByUserIdAndOffset(userId,offset,limit).get(0);//redisTemplate.opsForValue().set('newsKey',news);System.out.println('進入數據庫。。。。。。。。'); }else{ System.out.println('進入緩存。。。。。。。。。'); } return newsDAO.selectByUserIdAndOffset(userId,offset,limit); }}

然后是使用線程池在Controller里面對請求進行模擬:

@Controllerpublic class HomeController { @Autowired UserService userService; @Autowired NewsService newsService; //遇到的坑,如果不加method,頁面啟動不起來。 @RequestMapping(value = '/home',method = {RequestMethod.GET, RequestMethod.POST}) @ResponseBody public String index(Model model){ //這邊是可以讀出數據來的 //線程池------緩存穿透問題的復現 ExecutorService executorService = Executors.newFixedThreadPool(8*2); for(int i = 0;i < 50000;i++){ executorService.submit(new Runnable() {@Overridepublic 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(); }}

結果如圖:大量的請求進入數據庫,那么如何解決這個問題?

springboot中redis的緩存穿透問題實現

方法一、在方法上加鎖:

@Servicepublic class NewsService { @Autowired private NewsDAO newsDAO; //springboot自動初始化,不需要我們進行配置,直接注入到代碼中使用 @Autowired private RedisTemplate<Object,Object> redisTemplate; //第一種方式:方法加鎖 public synchronized List<News> getLatestNews(int userId,int offset,int limit){ //設置序列化方式,防止亂碼 redisTemplate.setKeySerializer(new StringRedisSerializer()); //第一步:查詢緩存 News news= (News) redisTemplate.opsForValue().get('newsKey'); //判斷是否存在緩存 if(null == news){//查詢數據庫news = newsDAO.selectByUserIdAndOffset(userId,offset,limit).get(0);//redisTemplate.opsForValue().set('newsKey',news);System.out.println('進入數據庫。。。。。。。。'); }else{ System.out.println('進入緩存。。。。。。。。。'); } return newsDAO.selectByUserIdAndOffset(userId,offset,limit); }}

直接在方法上加鎖,保證每次只有一個請求可以進入。但是這個方法存在一個缺陷,每次只有一個請求可以進入,請求處理的速度變得相當的慢,不利于系統的實時性。

方法二、使用雙重校驗鎖:

@Servicepublic class NewsService { @Autowired private NewsDAO newsDAO; //springboot自動初始化,不需要我們進行配置,直接注入到代碼中使用 @Autowired private RedisTemplate<Object,Object> redisTemplate; //第一種方式:方法加鎖 public /*synchronized*/ List<News> getLatestNews(int userId,int offset,int limit){ //設置序列化方式,防止亂碼 redisTemplate.setKeySerializer(new StringRedisSerializer()); //第一步:查詢緩存 News news= (News) redisTemplate.opsForValue().get('newsKey'); //判斷是否存在緩存 if(null == news){ //第二種方式:雙重檢測鎖 synchronized (this){//查詢數據庫news = newsDAO.selectByUserIdAndOffset(userId,offset,limit).get(0);//redisTemplate.opsForValue().set('newsKey',news);System.out.println('進入數據庫。。。。。。。。'); } }else{ System.out.println('進入緩存。。。。。。。。。'); } return newsDAO.selectByUserIdAndOffset(userId,offset,limit); }}

這個方法比較好,雖然不能保證只有一個請求請求數據庫,但是當第一批請求進來,第二批之后的所有請求全部會在緩存取數據。

到此這篇關于springboot中redis的緩存穿透問題實現的文章就介紹到這了,更多相關springboot redis緩存穿透內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Spring
相關文章:
主站蜘蛛池模板: 一级做a爰片久久毛片图片 一级做a爰片久久毛片看看 | 国产精品第 | 成人黄网大全在线观看 | 站长推荐国产午夜免费视频 | 亚洲国产综合精品中文第一区 | 欧美亚洲制服 | 亚洲第一毛片 | 国产精品一区欧美日韩制服 | 国产伦精品一区二区三区无广告 | 国产成人亚洲精品 | 成人黄色免费网站 | 中文字幕日韩一区 | 国产成 人 综合 亚洲网 | 国产一区二区三区国产精品 | 国产高清1024永久免费 | 国产精品亚洲一区二区在线观看 | 精品毛片免费看 | 日韩在线一区视频 | 黄色片在线观看网址 | 免费美女黄色 | 国产一区日韩二区欧美三 | www成人国产在线观看网站 | 66av99精品福利视频在线 | 小明看片成人永久在线观看 | 成人免费观看国产高清 | 66av99精品福利视频在线 | 国产丝袜脚| 国产精品福利短视在线播放频 | 99re在线精品视频 | 中文字幕日韩亚洲 | 一级做a免费视频观看网站 一级做a爰毛片 | 国产20岁美女一级毛片 | 国产你懂| 久久免费视频2 | 婷婷丁香五月中文字幕 | 肉色呻吟胯下丝袜高跟视频 | 亚洲国产精品成人午夜在线观看 | 成人网视频在线观看免费 | 国产精品爱久久久久久久小 | 日本二区免费一片黄2019 | 欧美影视一区 |