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

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

為了更好地說(shuō)明秒殺系統(tǒng)的實(shí)現(xiàn),我們這里以Java語(yǔ)言為例,介紹一些ZSet的使用:
“`Java
/**
* 刪除和插入ZSet型Redis數(shù)據(jù)
*/
String product = “product:uuid”;// 商品唯一ID,如JD商品ID、餓了么商品ID等
int stock = 100;// 商品庫(kù)存數(shù)
int expireSeconds = 180;// 商品超時(shí)時(shí)間
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 };
// 初始化庫(kù)存
redisTemplate.opsForZSet().add(product, String.valueOf(stock), 0);
// 自增庫(kù)存銷售量,3分鐘后失效
String productSold = “product:” + product + “:sold”;
redisTemplate.opsForValue().increment(productSold, 1);
redisTemplate.expire(productSold, expireSeconds, TimeUnit.SECONDS);
// 記錄每個(gè)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ù)過(guò)多結(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()) {
// 庫(kù)存售罄
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) {
// 庫(kù)存查詢失敗
continue;
}
int orderNum = Integer.valueOf(stockStr);
String sold = (String)operations.opsForValue().get(productSold);
if (sold == null || Integer.valueOf(sold) >= stock) {
// 店鋪超時(shí)或加入緩存失敗
continue;
}
if (orderNum
// 庫(kù)存售罄
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í)行減少庫(kù)存
operations.multi();
operations.opsForZSet().incrementScore(product, stockStr, -1);
operations.opsForValue().increment(productSold, 1);
Listlist = operations.exec();
if (list == null || list.isEmpty()) {
// 減庫(kù)存操作失敗
continue;
}
// 提交訂單輪詢
int result = itvPredix.pollUntilConditionMet(
() -> redisTemplate.opsForSet().isMember(“user:” + product + “:” + “08:order”, orderID),
120000L, 100L, null);
if (result == -1) {
// 訂閱超時(shí)
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 {
// 用戶超時(shí)
continue;
}
return true;
}
}
});
if (!flag) {
// 商品售罄
}
四、總結(jié)
通過(guò)以上示例代碼和流程圖,我們可以看出ZSet在秒殺系統(tǒng)中的重要性。它不僅可以保證商品數(shù)量的安全性,還可以支持高并發(fā)下多個(gè)請(qǐng)求的處理。但同時(shí)也有一些需要我們關(guān)注的方面,比如如何優(yōu)化ZSet的過(guò)期策略,提高ZSet的刪除效率等問(wèn)題,需要我們?cè)趯?shí)現(xiàn)中仔細(xì)考慮。
企業(yè)在實(shí)現(xiàn)秒殺系統(tǒng)時(shí),需要考慮的因素還有更多。比如如何保證系統(tǒng)的高可用、如何將Redis與其它數(shù)據(jù)庫(kù)進(jìn)行數(shù)據(jù)同步、如
成都服務(wù)器租用選創(chuàng)新互聯(lián),先試用再開通。
創(chuàng)新互聯(lián)(www.cdcxhl.com)提供簡(jiǎn)單好用,價(jià)格厚道的香港/美國(guó)云服務(wù)器和獨(dú)立服務(wù)器。物理服務(wù)器托管租用:四川成都、綿陽(yáng)、重慶、貴陽(yáng)機(jī)房服務(wù)器托管租用。
網(wǎng)站題目:Redis實(shí)現(xiàn)高性能秒殺ZSet有何魔力(redis的zset秒殺)
標(biāo)題網(wǎng)址:http://www.dlmjj.cn/article/ccssdhp.html


咨詢
建站咨詢
