我是靠谱客的博主 现实眼睛,这篇文章主要介绍解决高并发下的缓存穿透问题,现在分享给大家,希望可以做个参考。

  目前我们针对高并发场景,这种问题很常见,一般有如下两种解决方式:

  一:在方法上加上同步锁 synchronized(用的比较少)

  

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
//加同步锁,解决高并发下缓存穿透 @Test public synchronized void getMyUser(){ //字符串的序列化器 redis RedisSerializer redisSerializer = new StringRedisSerializer(); redisTemplate.setKeySerializer(redisSerializer); //查询缓存 MyUser myUser = (MyUser) redisTemplate.opsForValue().get("myUser"); if(null == myUser){ System.out.println("当缓存没有myUser时显示."); //缓存为空,查询数据库 myUser = myUserMapper.selectByPrimaryKey(1l); //把数据库查询出来的数据放入redis redisTemplate.opsForValue().set("myUser",myUser); } System.out.println(myUser); }

  二:使用双重检测锁(常用)

  

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public void getMyUser(){ //字符串的序列化器 redis RedisSerializer redisSerializer = new StringRedisSerializer(); redisTemplate.setKeySerializer(redisSerializer); //查询缓存 MyUser myUser = (MyUser) redisTemplate.opsForValue().get("myUser"); //双层检测锁 if(null == myUser){ synchronized(this){ myUser = (MyUser) redisTemplate.opsForValue().get("myUser"); if(null == myUser){ System.out.println("当缓存没有myUser时显示."); //缓存为空,查询数据库 myUser = myUserMapper.selectByPrimaryKey(1l); //把数据库查询出来的数据放入redis redisTemplate.opsForValue().set("myUser",myUser); } } } System.out.println(myUser); }

  三:进行高并发测试

  

复制代码
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
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
import com.ykmimi.job.bean.MyUser; import com.ykmimi.job.mapper.MyUserMapper; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.data.redis.core.RedisTemplate; import org.springframework.data.redis.serializer.RedisSerializer; import org.springframework.data.redis.serializer.StringRedisSerializer; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RestController; import javax.annotation.Resource; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; @RestController public class MyUserController { @Autowired private RedisTemplate<Object,Object> redisTemplate; @Resource private MyUserMapper myUserMapper; public Integer insertNew(){ return 0; } @RequestMapping("/getMyUserTest") public void getMyUserTest(){ //线程,该线程调用底层查询MyUser的方法 Runnable runnable = new Runnable() { @Override public void run() { getMyUser(); } }; //多线程测试一下缓存穿透的问题 ExecutorService executorService = Executors.newFixedThreadPool(25); for(int i=0;i<10000;i++){ executorService.submit(runnable); } } public void getMyUser(){ //字符串的序列化器 redis RedisSerializer redisSerializer = new StringRedisSerializer(); redisTemplate.setKeySerializer(redisSerializer); //查询缓存 MyUser myUser = (MyUser) redisTemplate.opsForValue().get("myUser"); //双层检测锁 if(null == myUser){ System.out.println("当缓存没有myUser时显示.没有进入同步锁"); synchronized(this){ myUser = (MyUser) redisTemplate.opsForValue().get("myUser"); if(null == myUser){ System.out.println("当缓存没有myUser时显示.已经进入同步锁"); //缓存为空,查询数据库 myUser = myUserMapper.selectByPrimaryKey(1l); //把数据库查询出来的数据放入redis redisTemplate.opsForValue().set("myUser",myUser); } } } System.out.println(myUser); } }

  住:线程池中不要特别大的线程

  

 

转载于:https://www.cnblogs.com/wangzhengyu/p/10621464.html

最后

以上就是现实眼睛最近收集整理的关于解决高并发下的缓存穿透问题的全部内容,更多相关解决高并发下内容请搜索靠谱客的其他文章。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(64)

评论列表共有 0 条评论

立即
投稿
返回
顶部