網(wǎng)站首頁 編程語言 正文
Redis 如何實(shí)現(xiàn)庫存扣減操作?如何防止商品被超賣?
基于數(shù)據(jù)庫單庫存 基于數(shù)據(jù)庫多庫存 基于redis 基于redis實(shí)現(xiàn)扣減庫存的具體實(shí)現(xiàn) 初始化庫存回調(diào)函數(shù)(IStockCallback) 扣減庫存服務(wù)(StockService)。
在日常開發(fā)中有很多地方都有類似扣減庫存的操作,比如電商系統(tǒng)中的商品庫存,抽獎(jiǎng)系統(tǒng)中的獎(jiǎng)品庫存等。
解決方案
1. 使用mysql數(shù)據(jù)庫
使用一個(gè)字段來存儲(chǔ)庫存,每次扣減庫存去更新這個(gè)字段。
2. 還是使用數(shù)據(jù)庫
但是將庫存分層多份存到多條記錄里面,扣減庫存的時(shí)候路由一下,這樣子增大了并發(fā)量,但是還是避免不了大量的去訪問數(shù)據(jù)庫來更新庫存。
3. 將庫存放到redis使用redis的incrby特性來扣減庫存。
分析
在上面的第一種和第二種方式都是基于數(shù)據(jù)來扣減庫存。
[基于數(shù)據(jù)庫單庫存]
第一種方式在所有請(qǐng)求都會(huì)在這里等待鎖,獲取鎖有去扣減庫存。在并發(fā)量不高的情況下可以使用,但是一旦并發(fā)量大了就會(huì)有大量請(qǐng)求阻塞在這里,導(dǎo)致請(qǐng)求超時(shí),進(jìn)而整個(gè)系統(tǒng)雪崩;而且會(huì)頻繁的去訪問數(shù)據(jù)庫,大量占用數(shù)據(jù)庫資源,所以在并發(fā)高的情況下這種方式不適用。
[基于數(shù)據(jù)庫多庫存]
第二種方式其實(shí)是第一種方式的優(yōu)化版本,在一定程度上提高了并發(fā)量,但是在還是會(huì)大量的對(duì)數(shù)據(jù)庫做更新操作大量占用數(shù)據(jù)庫資源。
基于數(shù)據(jù)庫來實(shí)現(xiàn)扣減庫存還存在的一些問題:
用數(shù)據(jù)庫扣減庫存的方式,扣減庫存的操作必須在一條語句中執(zhí)行,不能先selec在update,這樣在并發(fā)下會(huì)出現(xiàn)超扣的情況。如:
update number set x=x-1 where x > 0
- MySQL自身對(duì)于高并發(fā)的處理性能就會(huì)出現(xiàn)問題,一般來說,MySQL的處理性能會(huì)隨著并發(fā)thread上升而上升,但是到了一定的并發(fā)度之后會(huì)出現(xiàn)明顯的拐點(diǎn),之后一路下降,最終甚至?xí)葐蝨hread的性能還要差。
- 當(dāng)減庫存和高并發(fā)碰到一起的時(shí)候,由于操作的庫存數(shù)目在同一行,就會(huì)出現(xiàn)爭搶InnoDB行鎖的問題,導(dǎo)致出現(xiàn)互相等待甚至死鎖,從而大大降低MySQL的處理性能,最終導(dǎo)致前端頁面出現(xiàn)超時(shí)異常。
[基于redis]
針對(duì)上述問題的問題我們就有了第三種方案,將庫存放到緩存,利用redis的incrby特性來扣減庫存,解決了超扣和性能問題。但是一旦緩存丟失需要考慮恢復(fù)方案。比如抽獎(jiǎng)系統(tǒng)扣獎(jiǎng)品庫存的時(shí)候,初始庫存=總的庫存數(shù)-已經(jīng)發(fā)放的獎(jiǎng)勵(lì)數(shù),但是如果是異步發(fā)獎(jiǎng),需要等到MQ消息消費(fèi)完了才能重啟redis初始化庫存,否則也存在庫存不一致的問題。
基于redis實(shí)現(xiàn)扣減庫存的具體實(shí)現(xiàn)
- 我們使用redis的lua腳本來實(shí)現(xiàn)扣減庫存
- 由于是分布式環(huán)境下所以還需要一個(gè)分布式鎖來控制只能有一個(gè)服務(wù)去初始化庫存
- 需要提供一個(gè)回調(diào)函數(shù),在初始化庫存的時(shí)候去調(diào)用這個(gè)函數(shù)獲取初始化庫存
[初始化庫存回調(diào)函數(shù)(IStockCallback )]
/**
* 獲取庫存回調(diào)
* @author yuhao.wang
*/
public interface IStockCallback {
/**
* 獲取庫存
* @return
*/
int getStock();
}
[扣減庫存服務(wù)(StockService)]
/**
* 扣庫存
*
* @author yuhao.wang
*/
@Service
public class StockService {
Logger logger = LoggerFactory.getLogger(StockService.class);
/**
* 不限庫存
*/
public static final long UNINITIALIZED_STOCK = -3L;
/**
* Redis 客戶端
*/
@Autowired
private RedisTemplate<String, Object> redisTemplate;
/**
* 執(zhí)行扣庫存的腳本
*/
public static final String STOCK_LUA;
static {
/**
*
* @desc 扣減庫存Lua腳本
* 庫存(stock)-1:表示不限庫存
* 庫存(stock)0:表示沒有庫存
* 庫存(stock)大于0:表示剩余庫存
*
* @params 庫存key
* @return
* -3:庫存未初始化
* -2:庫存不足
* -1:不限庫存
* 大于等于0:剩余庫存(扣減之后剩余的庫存)
* redis緩存的庫存(value)是-1表示不限庫存,直接返回1
*/
StringBuilder sb = new StringBuilder();
sb.append("if (redis.call('exists', KEYS[1]) == 1) then");
sb.append(" local stock = tonumber(redis.call('get', KEYS[1]));");
sb.append(" local num = tonumber(ARGV[1]);");
sb.append(" if (stock == -1) then");
sb.append(" return -1;");
sb.append(" end;");
sb.append(" if (stock >= num) then");
sb.append(" return redis.call('incrby', KEYS[1], 0 - num);");
sb.append(" end;");
sb.append(" return -2;");
sb.append("end;");
sb.append("return -3;");
STOCK_LUA = sb.toString();
}
/**
* @param key 庫存key
* @param expire 庫存有效時(shí)間,單位秒
* @param num 扣減數(shù)量
* @param stockCallback 初始化庫存回調(diào)函數(shù)
* @return -2:庫存不足; -1:不限庫存; 大于等于0:扣減庫存之后的剩余庫存
*/
public long stock(String key, long expire, int num, IStockCallback stockCallback) {
long stock = stock(key, num);
// 初始化庫存
if (stock == UNINITIALIZED_STOCK) {
RedisLock redisLock = new RedisLock(redisTemplate, key);
try {
// 獲取鎖
if (redisLock.tryLock()) {
// 雙重驗(yàn)證,避免并發(fā)時(shí)重復(fù)回源到數(shù)據(jù)庫
stock = stock(key, num);
if (stock == UNINITIALIZED_STOCK) {
// 獲取初始化庫存
final int initStock = stockCallback.getStock();
// 將庫存設(shè)置到redis
redisTemplate.opsForValue().set(key, initStock, expire, TimeUnit.SECONDS);
// 調(diào)一次扣庫存的操作
stock = stock(key, num);
}
}
} catch (Exception e) {
logger.error(e.getMessage(), e);
} finally {
redisLock.unlock();
}
}
return stock;
}
/**
* 加庫存(還原庫存)
*
* @param key 庫存key
* @param num 庫存數(shù)量
* @return
*/
public long addStock(String key, int num) {
return addStock(key, null, num);
}
/**
* 加庫存
*
* @param key 庫存key
* @param expire 過期時(shí)間(秒)
* @param num 庫存數(shù)量
* @return
*/
public long addStock(String key, Long expire, int num) {
boolean hasKey = redisTemplate.hasKey(key);
// 判斷key是否存在,存在就直接更新
if (hasKey) {
return redisTemplate.opsForValue().increment(key, num);
}
Assert.notNull(expire,"初始化庫存失敗,庫存過期時(shí)間不能為null");
RedisLock redisLock = new RedisLock(redisTemplate, key);
try {
if (redisLock.tryLock()) {
// 獲取到鎖后再次判斷一下是否有key
hasKey = redisTemplate.hasKey(key);
if (!hasKey) {
// 初始化庫存
redisTemplate.opsForValue().set(key, num, expire, TimeUnit.SECONDS);
}
}
} catch (Exception e) {
logger.error(e.getMessage(), e);
} finally {
redisLock.unlock();
}
return num;
}
/**
* 獲取庫存
*
* @param key 庫存key
* @return -1:不限庫存; 大于等于0:剩余庫存
*/
public int getStock(String key) {
Integer stock = (Integer) redisTemplate.opsForValue().get(key);
return stock == null ? -1 : stock;
}
/**
* 扣庫存
*
* @param key 庫存key
* @param num 扣減庫存數(shù)量
* @return 扣減之后剩余的庫存【-3:庫存未初始化; -2:庫存不足; -1:不限庫存; 大于等于0:扣減庫存之后的剩余庫存】
*/
private Long stock(String key, int num) {
// 腳本里的KEYS參數(shù)
List<String> keys = new ArrayList<>();
keys.add(key);
// 腳本里的ARGV參數(shù)
List<String> args = new ArrayList<>();
args.add(Integer.toString(num));
long result = redisTemplate.execute(new RedisCallback<Long>() {
@Override
public Long doInRedis(RedisConnection connection) throws DataAccessException {
Object nativeConnection = connection.getNativeConnection();
// 集群模式和單機(jī)模式雖然執(zhí)行腳本的方法一樣,但是沒有共同的接口,所以只能分開執(zhí)行
// 集群模式
if (nativeConnection instanceof JedisCluster) {
return (Long) ((JedisCluster) nativeConnection).eval(STOCK_LUA, keys, args);
}
// 單機(jī)模式
else if (nativeConnection instanceof Jedis) {
return (Long) ((Jedis) nativeConnection).eval(STOCK_LUA, keys, args);
}
return UNINITIALIZED_STOCK;
}
});
return result;
}
}
[調(diào)用]
/**
* @author yuhao.wang
*/
@RestController
public class StockController {
@Autowired
private StockService stockService;
@RequestMapping(value = "stock", produces = MediaType.APPLICATION_JSON_UTF8_VALUE)
public Object stock() {
// 商品ID
long commodityId = 1;
// 庫存ID
String redisKey = "redis_key:stock:" + commodityId;
long stock = stockService.stock(redisKey, 60 * 60, 2, () -> initStock(commodityId));
return stock >= 0;
}
/**
* 獲取初始的庫存
*
* @return
*/
private int initStock(long commodityId) {
// TODO 這里做一些初始化庫存的操作
return 1000;
}
@RequestMapping(value = "getStock", produces = MediaType.APPLICATION_JSON_UTF8_VALUE)
public Object getStock() {
// 商品ID
long commodityId = 1;
// 庫存ID
String redisKey = "redis_key:stock:" + commodityId;
return stockService.getStock(redisKey);
}
@RequestMapping(value = "addStock", produces = MediaType.APPLICATION_JSON_UTF8_VALUE)
public Object addStock() {
// 商品ID
long commodityId = 2;
// 庫存ID
String redisKey = "redis_key:stock:" + commodityId;
return stockService.addStock(redisKey, 2);
}
}
結(jié)語
原文鏈接:https://blog.csdn.net/qq_44866828/article/details/125500046
相關(guān)推薦
- 2022-10-10 C++超詳細(xì)分析type_traits_C 語言
- 2021-12-02 docker容器時(shí)區(qū)錯(cuò)誤問題_docker
- 2022-04-24 C語言的abs()函數(shù)和div()函數(shù)你了解嗎_C 語言
- 2022-07-17 C#編程報(bào)錯(cuò)System.InvalidOperationException問題及解決_C#教程
- 2022-08-31 Python的三種主要模塊介紹_python
- 2022-05-06 Sqlite 刪除數(shù)據(jù)后為什么文件大小不變
- 2022-07-25 SQL?Server系統(tǒng)函數(shù)介紹_MsSql
- 2022-05-06 詳析Python面向?qū)ο笾械睦^承_python
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支