日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > 数据库 >内容正文

数据库

解决Redis超卖问题

發(fā)布時間:2025/3/21 数据库 17 豆豆
生活随笔 收集整理的這篇文章主要介紹了 解决Redis超卖问题 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

第一種:使用synchronized(只適用于單個tomcat)

@Autowired private StringRedisTemplate stringRedisTemplate; @GetMapping("/deduct")public String toDeduct(){synchronized (this) {int stock = Integer.parseInt(stringRedisTemplate.opsForValue().get("stock"));if (stock > 0) {Long stock1 = stringRedisTemplate.opsForValue().decrement("stock");System.out.println("扣減成功,剩余庫存:" + stock1);}else{System.out.println("扣減失敗,庫存不足");}return "end";}}

第二種:使用SETNX命令(多臺tomcat)

@GetMapping("/deduct_stock")public String deductStock() throws InterruptedException {String lockKey = "lockKey";//設計唯一key值防止鎖失效問題String clientId = UUID.randomUUID().toString();//設計鎖超時時間,防止服務器宕機時鎖沒有釋放掉(finally語句沒有執(zhí)行)Boolean result = stringRedisTemplate.opsForValue().setIfAbsent(lockKey, clientId, 10, TimeUnit.SECONDS);//jedis.setnx(key,value);//若鍵 key 已經(jīng)存在, 則 SETNX 命令不做任何動作。result==falseif (!result) {return "正在排隊。。。";}try {int stock = Integer.parseInt(stringRedisTemplate.opsForValue().get("stock"));if (stock > 0) {Long realStock = stringRedisTemplate.opsForValue().decrement("stock");System.out.println("扣減成功,剩余庫存:" + realStock);}else{System.out.println("扣減失敗,庫存不足");}} finally {//防止鎖失效問題,在多線程的情況下,每個線程只釋放自己創(chuàng)建的鎖,線程之間互不干預。if(clientId.equals(stringRedisTemplate.opsForValue().get(lockKey))) {stringRedisTemplate.delete(lockKey);}}return "end";}

第三種:使用Redission框架(原理跟第二種一樣)

1、配置config類

@Configuration public class RedissonConfig {@Value("${spring.redis.host}")private String host;@Value("${spring.redis.port}")private String port;@Value("${spring.redis.password}")private String password;@Beanpublic RedissonClient getRedisson(){Config config = new Config();config.useSingleServer().setAddress("redis://" + host + ":" + port).setPassword(password);//添加主從配置 // config.useMasterSlaveServers().setMasterAddress("").setPassword("").addSlaveAddress(new String[]{"",""});return Redisson.create(config);} } @GetMapping("/deduct_stock1")public String deductStock1(){String lockKey = "lockKey";RLock redissonLock = redisson.getLock(lockKey);try {redissonLock.lock();int stock = Integer.parseInt(stringRedisTemplate.opsForValue().get("stock"));if (stock > 0) {Long realStock = stringRedisTemplate.opsForValue().decrement("stock");System.out.println("扣減成功,剩余庫存:" + realStock);}else{System.out.println("扣減失敗,庫存不足");}} finally {redissonLock.unlock();}return "end";}

Redission原理:

Redission使用了Lua 腳本,實現(xiàn)原理跟原理跟第二種差不多。
當業(yè)務代碼處理時間很長,像mysql慢查詢等等,可以在try代碼塊中每隔10秒檢查是否還持有鎖,如果持有則延長鎖的時間,防止還沒執(zhí)行完業(yè)務代碼“l(fā)ockKey”就到了失效時間。


測試環(huán)境搭建

Redis命令大全

設置Redis密碼

pom:

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId></dependency><dependency><groupId>org.redisson</groupId><artifactId>redisson</artifactId><version>3.5.7</version></dependency>

application.properties:

server.port=8090 # redis spring.redis.host=127.0.0.1 spring.redis.port=6379 spring.redis.password=888888 spring.redis.jedis.pool.max-active=500 spring.redis.jedis.pool.max-idle=1000 spring.redis.jedis.pool.max-wait=6000ms spring.redis.jedis.pool.min-idle=4

Nginx負載均衡:

upstream redislock{server 127.0.0.1:8080 weight = 1;server 127.0.0.1:8090 weight = 1;}server {listen 80;server_name localhost;location / {root html;index index.html index.htm;proxy_pass http://redislock;}}

Jmeter壓力測試


發(fā)送http請求:

設置發(fā)送地址:

表示在0秒內(nèi)發(fā)送200個請求,處理完后再發(fā)送200 * 3次,總計200 * 4次請求:

結果集:

測試結果:


沒有出現(xiàn)超賣問題

總結

以上是生活随笔為你收集整理的解决Redis超卖问题的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。