新聞中心
Redis實現(xiàn)高性能秒殺:zset有何魔力?

秒殺活動作為一種營銷手段,已經(jīng)受到越來越多商家的重視。而如何保證在短時間內(nèi)完成大量用戶的請求,保障用戶和系統(tǒng)的體驗,成為了一個難題。在這樣的背景下,Redis作為一個高性能、高可用、支持多種數(shù)據(jù)結(jié)構(gòu)的內(nèi)存數(shù)據(jù)庫,正成為越來越多企業(yè)選擇的解決方案。在Redis中,ZSet(有序集合)的獨特屬性使其成為實現(xiàn)高性能秒殺的選擇之一。
一、什么是ZSet
Redis是一個鍵值對存儲系統(tǒng),其中又包含五種基本數(shù)據(jù)類型:string(字符串)、Hash(哈希表)、List(列表)、Set(集合)和ZSet(有序集合)。ZSet類似于Set,它們都是不允許出現(xiàn)重復(fù)元素的容器。與Set不同的是,ZSet中的元素是可排序的,且每個元素都會關(guān)聯(lián)一個權(quán)重值score,Redis會根據(jù)score將元素從小到大排序。ZSet的主要操作包括插入元素、刪除元素和獲取元素排名和權(quán)重值等。
二、ZSet在秒殺中的應(yīng)用
在秒殺的場景中,我們需要解決兩個問題:
1. 如何保證商品數(shù)量的安全性,防止售罄后用戶還能下單?
2. 如何保證用戶在短時間內(nèi)完成下單的請求,并避免重復(fù)提交?
針對上述問題,使用ZSet實現(xiàn)秒殺有以下優(yōu)勢:
1. ZSet可以實現(xiàn)商品數(shù)量的安全性。在Redis中,我們可以通過ZSet的插入元素操作,將商品的庫存作為score關(guān)聯(lián)到商品的ID作為元素,這樣即可保證商品數(shù)量的安全性。當(dāng)用戶下單時,可以通過ZSet的刪除元素操作,將商品的庫存score減1,以及將訂單信息作為另一個ZSet的元素插入,直到庫存為0時,ZSet中該商品ID的元素將被刪除,再次下單則無法成功。
2. ZSet可以保證短時間內(nèi)完成大量用戶請求。在Redis中,我們可以使用ZSet的分值排序功能,在查詢秒殺商品庫存時,將ZSet中的所有Elem刷新到本地。如此一來,當(dāng)庫存有余量時,用戶請求可以被快速響應(yīng);而當(dāng)庫存售罄時,用戶請求則會排隊等待,避免重復(fù)下單。
三、ZSet的使用
針對上述問題,我們對秒殺系統(tǒng)的實現(xiàn)可以畫出大概的流程圖如下:

為了更好地說明秒殺系統(tǒng)的實現(xiàn),我們這里以Java語言為例,介紹一些ZSet的使用:
“`Java
/**
* 刪除和插入ZSet型Redis數(shù)據(jù)
*/
String product = “product:uuid”;// 商品唯一ID,如JD商品ID、餓了么商品ID等
int stock = 100;// 商品庫存數(shù)
int expireSeconds = 180;// 商品超時時間
int limit = 10;// 最大提交次數(shù)
String[] orders = { “5dd34e5b-cd18-4afa-b639-a58eabe7883f”, “613c18d8-b496-453f-855e-ba1a46dfc0d7”,
“755f42b7-8475-4a13-a59f-2295b5e5e746” };
double[] scores = { 3.0, 2.0, 1.0 };
// 初始化庫存
redisTemplate.opsForZSet().add(product, String.valueOf(stock), 0);
// 自增庫存銷售量,3分鐘后失效
String productSold = “product:” + product + “:sold”;
redisTemplate.opsForValue().increment(productSold, 1);
redisTemplate.expire(productSold, expireSeconds, TimeUnit.SECONDS);
// 記錄每個IP的提交次數(shù),10次后被禁止提交
String userLimit = “userLimit:” + product + “:ip”;
redisTemplate.opsForValue().setIfAbsent(userLimit, “0”);
redisTemplate.expire(userLimit, expireSeconds, TimeUnit.SECONDS);
Long count = redisTemplate.opsForValue().increment(userLimit, 1);
if (count > limit) {
// 返回提交次數(shù)過多結(jié)果
}
// 秒殺下單
String orderID = UUID.randomUUID().toString();
Boolean flag = redisTemplate.execute(new SessionCallback() {
@SuppressWarnings(“unchecked”)
@Override
public Boolean execute(RedisOperations operations) throws DataAccessException {
while (true) {
operations.watch(product);
Set> stringSet = operations.opsForZSet().rangeByScoreWithScores(product, 0, stock);
if (stringSet == null || stringSet.isEmpty()) {
// 庫存售罄
return false;
}
Iterator> iterator = stringSet.iterator();
String stockStr = null;
Double score = null;
if (iterator.hasNext()) {
ZSetOperations.TypedTuple typedTuple = iterator.next();
stockStr = typedTuple.getValue();
score = typedTuple.getScore();
}
if (stockStr == null || score == null) {
// 庫存查詢失敗
continue;
}
int orderNum = Integer.valueOf(stockStr);
String sold = (String)operations.opsForValue().get(productSold);
if (sold == null || Integer.valueOf(sold) >= stock) {
// 店鋪超時或加入緩存失敗
continue;
}
if (orderNum
// 庫存售罄
return false;
}
// 插入秒殺訂單
ZSetOperations.TypedTuple order = operations.opsForZSet().add(product + “:order”, orderID,
score);
if (order == null) {
// 插入訂單失敗
continue;
}
// 保存下單成功的訂單ID
redisTemplate.opsForSet().add(“user:” + product + “:” + “08:order”, orderID);
// 事務(wù)執(zhí)行減少庫存
operations.multi();
operations.opsForZSet().incrementScore(product, stockStr, -1);
operations.opsForValue().increment(productSold, 1);
Listlist = operations.exec();
if (list == null || list.isEmpty()) {
// 減庫存操作失敗
continue;
}
// 提交訂單輪詢
int result = itvPredix.pollUntilConditionMet(
() -> redisTemplate.opsForSet().isMember(“user:” + product + “:” + “08:order”, orderID),
120000L, 100L, null);
if (result == -1) {
// 訂閱超時
continue;
} else if (result == 1) {
// 訂閱成功
Map orderDetl = new HashMap();
orderDetl.put(“orderID”, orderID);
orderDetl.put(“product”, product);
orderDetl.put(“createAt”, System.currentTimeMillis());
// 返回秒殺成功結(jié)果,并推送消息到MQ
} else {
// 用戶超時
continue;
}
return true;
}
}
});
if (!flag) {
// 商品售罄
}
四、總結(jié)
通過以上示例代碼和流程圖,我們可以看出ZSet在秒殺系統(tǒng)中的重要性。它不僅可以保證商品數(shù)量的安全性,還可以支持高并發(fā)下多個請求的處理。但同時也有一些需要我們關(guān)注的方面,比如如何優(yōu)化ZSet的過期策略,提高ZSet的刪除效率等問題,需要我們在實現(xiàn)中仔細(xì)考慮。
企業(yè)在實現(xiàn)秒殺系統(tǒng)時,需要考慮的因素還有更多。比如如何保證系統(tǒng)的高可用、如何將Redis與其它數(shù)據(jù)庫進行數(shù)據(jù)同步、如
成都服務(wù)器租用選創(chuàng)新互聯(lián),先試用再開通。
創(chuàng)新互聯(lián)(www.cdcxhl.com)提供簡單好用,價格厚道的香港/美國云服務(wù)器和獨立服務(wù)器。物理服務(wù)器托管租用:四川成都、綿陽、重慶、貴陽機房服務(wù)器托管租用。
新聞名稱:Redis實現(xiàn)高性能秒殺ZSet有何魔力(redis的zset秒殺)
標(biāo)題來源:http://fisionsoft.com.cn/article/ccssdhp.html


咨詢
建站咨詢
