Redsi常见问题
缓存在高平发和安全压力下的一些问题
缓存击穿
是某一个热点key在高并发访问的情况下,突然失效,导致大量的并发大金mysql数据库的情况
缓存穿透
是利用redis和mysql的机制(redis缓存一旦不存在,就访问mysql),直接让过缓存访问mysql,而制造的db请求压力
一般在代码中防止
解决: 为防止缓存穿透,将null或者空字符串设置给redis
缓存雪崩
缓存是采用了相同的过期时间,导致缓存在某一时刻同时全部失效,导致的db崩溃
解决:设计不同的缓存失效时间
如何解决缓存击穿的问题 ?
分布式锁
data:image/s3,"s3://crabby-images/a6077/a607737fae96ae8ff60bfe6cecbeef6eb79bda21" alt=""
穿透:利用不存在的key去攻击mysql数据库
雪崩:缓存中的很多key失效,导致数据库负载过重宕机
击穿:在正常的访问情况下,如果缓存失效,如果保护mysql,重启缓存的过程
使用redis数据库的分布式锁,解决mysql的访问压力问题
第一种分布式锁:redis自带一个分布式锁,set px nx
data:image/s3,"s3://crabby-images/f0bdf/f0bdfc85470a1f5c594dc6438881b4e5152ea2a6" alt=""
data:image/s3,"s3://crabby-images/e9677/e9677d5765770c680d5d43b256dab6a614716b55" alt=""
用完之后需要删除,不然别人不能访问。
data:image/s3,"s3://crabby-images/dcdb0/dcdb057cdde21202e41b12c54498cc7d7c1d99bd" alt=""
错误自旋代码(B为孤儿线程)
data:image/s3,"s3://crabby-images/eb201/eb2019139c6af692ade99257a477654758f2d23a" alt=""
正确自旋代码
data:image/s3,"s3://crabby-images/bd442/bd44238adb242c84911b83e1568cb70654548a8f" alt=""
问题1 如果在redis中的锁已经过期了,然后锁过期的那个请求又执行完毕,回来删锁,删除了其他线程的锁,怎么办?
问题2 如果碰巧在查询redis锁还没删除的时候,正在网络传输时,锁过期了,怎么办?
1
2
|
String script ="if redis.call('get', KEYS[1]) == ARGV[1] then return redis.call('del', KEYS[1]) else return 0 end";
jedis.eval(script, Collections.singletonList("lock"),Collections.singletonList(token));
|
data:image/s3,"s3://crabby-images/621ae/621ae27013ed00d8538bdaf3f9c1a44d77f9b629" alt=""
第二种分布式锁:redisson框架,一个redis的带有juc的lock功能的客户端(既有jedis的功能,又有juc的功能)
data:image/s3,"s3://crabby-images/17c7c/17c7ce6dda07d75ee9d6bc6dbeae9fc42d38dec5" alt=""
整合
引入pom
1
2
3
4
5
6
|
<!-- https://mvnrepository.com/artifact/org.redisson/redisson -->
<dependency>
<groupId>org.redisson</groupId>
<artifactId>redisson</artifactId>
<version>3.10.5</version>
</dependency>
|
配置
1
2
|
spring.redis.host=192.168.159.130
spring.redis.port=6379
|
配置类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
@Configuration
public class GmallRedissonConfig {
@Value("${spring.redis.host}")
private String host;
@Value("${spring.redis.port}")
private String port;
@Bean
public RedissonClient redissonClient(){
Config config = new Config();
config.useSingleServer().setAddress("redis://"+host+":"+port);
RedissonClient redisson = Redisson.create(config);
return redisson;
}
}
|
Redisson实现了juc的lock锁,并且可以在分布式的redis环境下使用
data:image/s3,"s3://crabby-images/76626/76626b1a3cf4cebad4bd3600148a1461fd9f6ca5" alt=""
data:image/s3,"s3://crabby-images/95147/9514703e83505ae379ea96958604d9c0ecdc991c" alt=""
添加gmall-redisson-text
把需要的依赖添加的pom上
data:image/s3,"s3://crabby-images/8e73b/8e73b4e74df38ca7acb99e39888ef8c007d1dcae" alt=""
配置application.properties
1
2
3
4
|
# 服务端口
server.port=8082
# 日志级别
logging.level.root=info
|
RedissonController
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
|
package com.atguigu.gmallredisson.redissonTest;
import com.atguigu.gmall.util.RedisUtil;
import org.apache.commons.lang3.StringUtils;
import org.redisson.api.RLock;
import org.redisson.api.RedissonClient;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import redis.clients.jedis.Jedis;
@Controller
public class RedissonController {
@Autowired
RedisUtil redisUtil;
@Autowired
RedissonClient redissonClient;
@RequestMapping("testRedisson")
@ResponseBody
public String testRedisson(){
Jedis jedis = redisUtil.getJedis();
RLock lock = redissonClient.getLock("lock");// 声明锁
lock.lock();//上锁
try {
String v = jedis.get("k");
if (StringUtils.isBlank(v)) {
v = "1";
}
System.out.println("->" + v);
jedis.set("k", (Integer.parseInt(v) + 1) + "");
}finally {
jedis.close();
lock.unlock();// 解锁
}
return "success";
}
}
|
设置非单例模式启动
data:image/s3,"s3://crabby-images/4a820/4a820660649f04b3b0919aa0740aa1819a845d3f" alt=""
点击取消单一示例
data:image/s3,"s3://crabby-images/36e57/36e57c4327393206fd85e72d918577149a019386" alt=""
启动3个示例,端口分别为8080,8081,8082
data:image/s3,"s3://crabby-images/c4905/c490591be3d9ef428e9fc38f51e80414ce2c7b89" alt=""
配置nginx
data:image/s3,"s3://crabby-images/09c66/09c66eef8338ca230b9903330286321ccb636fd2" alt=""
下载安装apache测试工具(apache)
1 下载地址
https://www.apachehaus.com/cgi-bin/download.plx
data:image/s3,"s3://crabby-images/9b994/9b994a632f0e5246a377e25c10d325933dd6df2c" alt=""
2 安装即解压
data:image/s3,"s3://crabby-images/641c6/641c62f9fbf221bdef467abb05c761b65f612064" alt=""
3 修改apache服务的配置文件(服务器的根目录)
data:image/s3,"s3://crabby-images/5b1f5/5b1f56436d23d841b8781db35abfa7ab8c2c7a77" alt=""
修改服务的根目录路径:
data:image/s3,"s3://crabby-images/248ee/248ee2846ed0d609b81a11cedcabb52c83756c02" alt=""
4 启动服务
查看443端口是否被占用
1
|
D:\ap\Apache24\bin>netstat -ano | findstr "443"
|
data:image/s3,"s3://crabby-images/13878/13878977a25128a3daff84746c690b80b9c0200e" alt=""
没有被占用,启动httpd.exe
1
|
D:\ap\Apache24\bin>httpd.exe
|
5 压测命令(另起cmd输入命令)
D:\apache24\bin>ab -c 200 -n 1000 http:nginx负载均衡/压力方法
测试:
1
|
D:\ap\Apache24\bin>ab -c 100 -n 10000 http://www.der-matech.com.cn/
|
data:image/s3,"s3://crabby-images/37693/3769362b2028f6b58c0dabcaeb740c874d18742e" alt=""
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
public String testRedisson(){
Jedis jedis = redisUtil.getJedis();
RLock lock = redissonClient.getLock("lock");// 声明锁
lock.lock();//上锁
try {
String v = jedis.get("k");
if (StringUtils.isBlank(v)) {
v = "1";
}
System.out.println("->" + v);
jedis.set("k", (Integer.parseInt(v) + 1) + "");
}finally {
jedis.close();
lock.unlock();// 解锁
}
return "success";
}
|