springboot中redis的缓存穿透问题实现

2025-05-29 0 56

什么是缓存穿透问题??

我们使用redis是为了减少数据库的压力,让尽量多的请求去承压能力比较大的redis,而不是数据库。但是高并发条件下,可能会在redis还没有缓存的时候,大量的请求同时进入,导致一大批的请求直奔数据库,而不会经过redis。使用代码模拟缓存穿透问题如下:

首先是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
@Service

public 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里面对请求进行模拟:

?

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
@Controller

public 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() {

@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();

}

}

结果如图:大量的请求进入数据库,那么如何解决这个问题?

springboot中redis的缓存穿透问题实现

方法一、在方法上加锁:

?

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
@Service

public 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);

}

}

直接在方法上加锁,保证每次只有一个请求可以进入。但是这个方法存在一个缺陷,每次只有一个请求可以进入,请求处理的速度变得相当的慢,不利于系统的实时性。

方法二、使用双重校验锁:

?

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
@Service

public 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缓存穿透内容请搜索快网idc以前的文章或继续浏览下面的相关文章希望大家以后多多支持快网idc!

原文链接:https://www.cnblogs.com/quintanliu/p/13429690.html

收藏 (0) 打赏

感谢您的支持,我会继续努力的!

打开微信/支付宝扫一扫,即可进行扫码打赏哦,分享从这里开始,精彩与您同在
点赞 (0)

声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。

快网idc优惠网 建站教程 springboot中redis的缓存穿透问题实现 https://www.kuaiidc.com/108861.html

相关文章

猜你喜欢
发表评论
暂无评论