在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

如何實現一個秒殺系統

Android編程精選 ? 來源:Android編程精選 ? 作者:Android編程精選 ? 2022-09-15 09:56 ? 次閱讀

實現一個秒殺系統,采用spring boot 2.x + mybatis+ redis + swagger2 + lombok實現。

先說說基本流程,就是提供一個秒殺接口,然后針對秒殺接口進行限流,限流的方式目前我實現了兩種,上次實現的是累計計數方式,這次還有這個功能,并且我增加了令牌桶方式的lua腳本進行限流。

然后不被限流的數據進來之后,加一把分布式鎖,獲取分布式鎖之后就可以對數據庫進行操作了。直接操作數據庫的方式可以,但是速度會比較慢,咱們直接通過一個初始化接口,將庫存數據放到緩存中,然后對緩存中的數據進行操作。

寫庫的操作采用異步方式,實現的方式就是將操作好的數據放入到隊列中,然后由另一個線程對隊列進行消費。當然,也可以將數據直接寫入mq中,由另一個線程進行消費,這樣也更穩妥。

好了,看一下項目的基本結構:

e8a4b096-3421-11ed-ba43-dac502259ad0.jpg

看一下入口controller類,入口類有兩個方法,一個是初始化訂單的方法,即秒殺開始的時候,秒殺接口才會有效,這個方法可以采用定時任務自動實現也可以。

初始化后就可以調用placeOrder的方法了。在placeOrder上面有個自定義的注解DistriLimitAnno,這個是我在上篇文章寫的,用作限流使用。

采用的方式目前有兩種,一種是使用計數方式限流,一種方式是令牌桶,上次使用了計數,咱們這次采用令牌桶方式實現。

package com.hqs.flashsales.controller;

import com.hqs.flashsales.annotation.DistriLimitAnno;import com.hqs.flashsales.aspect.LimitAspect;import com.hqs.flashsales.lock.DistributedLock;import com.hqs.flashsales.limit.DistributedLimit;import com.hqs.flashsales.service.OrderService;import lombok.extern.slf4j.Slf4j;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.data.redis.core.RedisTemplate;import org.springframework.data.redis.core.script.RedisScript;import org.springframework.stereotype.Controller;import org.springframework.web.bind.annotation.GetMapping;import org.springframework.web.bind.annotation.PostMapping;import org.springframework.web.bind.annotation.ResponseBody;

import javax.annotation.Resource;import java.util.Collections;

/** * @author huangqingshi * @Date 2019-01-23 */@Slf4j@Controllerpublic class FlashSaleController {

@Autowired OrderService orderService; @Autowired DistributedLock distributedLock; @Autowired LimitAspect limitAspect; //注意RedisTemplate用的String,String,后續所有用到的key和value都是String的 @Autowired RedisTemplate redisTemplate;

private static final String LOCK_PRE = "LOCK_ORDER";

@PostMapping("/initCatalog") @ResponseBody public String initCatalog() { try { orderService.initCatalog(); } catch (Exception e) { log.error("error", e); }

return "init is ok"; }

@PostMapping("/placeOrder") @ResponseBody @DistriLimitAnno(limitKey = "limit", limit = 100, seconds = "1") public Long placeOrder(Long orderId) { Long saleOrderId = 0L; boolean locked = false; String key = LOCK_PRE + orderId; String uuid = String.valueOf(orderId); try { locked = distributedLock.distributedLock(key, uuid, "10" ); if(locked) { //直接操作數據庫// saleOrderId = orderService.placeOrder(orderId); //操作緩存 異步操作數據庫 saleOrderId = orderService.placeOrderWithQueue(orderId); } log.info("saleOrderId:{}", saleOrderId); } catch (Exception e) { log.error(e.getMessage()); } finally { if(locked) { distributedLock.distributedUnlock(key, uuid); } } return saleOrderId; }

}


令牌桶的方式比直接計數更加平滑,直接計數可能會瞬間達到最高值,令牌桶則把最高峰給削掉了,令牌桶的基本原理就是有一個桶裝著令牌,然后又一隊人排隊領取令牌,領到令牌的人就可以去做做自己想做的事情了,沒有領到令牌的人直接就走了(也可以重新排隊)。

發令牌是按照一定的速度發放的,所以這樣在多人等令牌的時候,很多人是拿不到的。當桶里邊的令牌在一定時間內領完后,則沒有令牌可領,都直接走了。如果過了一定的時間之后可以再次把令牌桶裝滿供排隊的人領。

基本原理是這樣的,看一下腳本簡單了解一下,里邊有一個key和四個參數,第一個參數是獲取一個令牌桶的時間間隔,第二個參數是重新填裝令牌的時間(精確到毫秒),第三個是令牌桶的數量限制,第四個是隔多長時間重新填裝令牌桶。

-- bucket namelocal key = KEYS[1]-- token generate intervallocal intervalPerPermit = tonumber(ARGV[1])-- grant timestamplocal refillTime = tonumber(ARGV[2])-- limit token countlocal limit = tonumber(ARGV[3])-- ratelimit time periodlocal interval = tonumber(ARGV[4])

local counter = redis.call('hgetall', key)

if table.getn(counter) == 0 then -- first check if bucket not exists, if yes, create a new one with full capacity, then grant access redis.call('hmset', key, 'lastRefillTime', refillTime, 'tokensRemaining', limit - 1) -- expire will save memory redis.call('expire', key, interval) return 1elseif table.getn(counter) == 4 then -- if bucket exists, first we try to refill the token bucket local lastRefillTime, tokensRemaining = tonumber(counter[2]), tonumber(counter[4]) local currentTokens if refillTime > lastRefillTime then -- check if refillTime larger than lastRefillTime. -- if not, it means some other operation later than this call made the call first. -- there is no need to refill the tokens. local intervalSinceLast = refillTime - lastRefillTime if intervalSinceLast > interval then currentTokens = limit redis.call('hset', key, 'lastRefillTime', refillTime) else local grantedTokens = math.floor(intervalSinceLast / intervalPerPermit) if grantedTokens > 0 then -- ajust lastRefillTime, we want shift left the refill time. local padMillis = math.fmod(intervalSinceLast, intervalPerPermit) redis.call('hset', key, 'lastRefillTime', refillTime - padMillis) end currentTokens = math.min(grantedTokens + tokensRemaining, limit) end else -- if not, it means some other operation later than this call made the call first. -- there is no need to refill the tokens. currentTokens = tokensRemaining end

assert(currentTokens >= 0)

if currentTokens == 0 then -- we didn't consume any keys redis.call('hset', key, 'tokensRemaining', currentTokens) return 0 else -- we take 1 token from the bucket redis.call('hset', key, 'tokensRemaining', currentTokens - 1) return 1 endelse error("Size of counter is " .. table.getn(counter) .. ", Should Be 0 or 4.")end

看一下調用令牌桶lua的JAVA代碼,也比較簡單:

public Boolean distributedRateLimit(String key, String limit, String seconds) {    Long id = 0L;    long intervalInMills = Long.valueOf(seconds) * 1000;    long limitInLong = Long.valueOf(limit);    long intervalPerPermit = intervalInMills / limitInLong;//    Long refillTime = System.currentTimeMillis();//    log.info("調用redis執行lua腳本, {} {} {} {} {}", "ratelimit", intervalPerPermit, refillTime,//        limit, intervalInMills);    try {       id = redisTemplate.execute(rateLimitScript, Collections.singletonList(key),          String.valueOf(intervalPerPermit), String.valueOf(System.currentTimeMillis()),          String.valueOf(limitInLong), String.valueOf(intervalInMills));    } catch (Exception e) {      log.error("error", e);    }

if(id == 0L) { return false; } else { return true; } }


創建兩張簡單表,一個庫存表,一個是銷售訂單表:

CREATE TABLE `catalog` ( `id` int(11) unsigned NOT NULL AUTO_INCREMENT, `name` varchar(50) NOT NULL DEFAULT '' COMMENT '名稱', `total` int(11) NOT NULL COMMENT '庫存', `sold` int(11) NOT NULL COMMENT '已售', `version` int(11) NULL COMMENT '樂觀鎖,版本號', PRIMARY KEY (`id`)) ENGINE=InnoDB DEFAULT CHARSET=utf8;

CREATE TABLE `sales_order` ( `id` int(11) unsigned NOT NULL AUTO_INCREMENT, `cid` int(11) NOT NULL COMMENT '庫存ID', `name` varchar(30) NOT NULL DEFAULT '' COMMENT '商品名稱', `create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '創建時間', PRIMARY KEY (`id`)) ENGINE=InnoDB DEFAULT CHARSET=utf8;


基本已經準備完畢,然后啟動程序,打開swagger(http://localhost:8080/swagger-ui.html#),執行初始化方法initCatalog:

e8b3ac2c-3421-11ed-ba43-dac502259ad0.jpg

日志里邊會輸出初始化的記錄內容,初始化庫存為1000:

e8be7300-3421-11ed-ba43-dac502259ad0.png

初始化執行的方法,十分簡單,寫到緩存中。

@Override  public void initCatalog() {    Catalog catalog = new Catalog();    catalog.setName("mac");    catalog.setTotal(1000L);    catalog.setSold(0L);    catalogMapper.insertCatalog(catalog);    log.info("catalog:{}", catalog);    redisTemplate.opsForValue().set(CATALOG_TOTAL + catalog.getId(), catalog.getTotal().toString());    redisTemplate.opsForValue().set(CATALOG_SOLD + catalog.getId(), catalog.getSold().toString());    log.info("redis value:{}", redisTemplate.opsForValue().get(CATALOG_TOTAL + catalog.getId()));    handleCatalog();  }

我寫了一個測試類,啟動3000個線程,然后去進行下單請求:

package com.hqs.flashsales;

import lombok.extern.slf4j.Slf4j;import org.junit.Test;import org.junit.runner.RunWith;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.boot.test.context.SpringBootTest;import org.springframework.boot.test.web.client.TestRestTemplate;import org.springframework.test.context.junit4.SpringRunner;import org.springframework.util.LinkedMultiValueMap;import org.springframework.util.MultiValueMap;

import java.util.concurrent.TimeUnit;

@Slf4j@RunWith(SpringRunner.class)@SpringBootTest(classes = FlashsalesApplication.class, webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)public class FlashSalesApplicationTests {

@Autowired private TestRestTemplate testRestTemplate;

@Test public void flashsaleTest() { String url = "http://localhost:8080/placeOrder"; for(int i = 0; i < 3000; i++) { ? ? ? ? ? ?try { ? ? ? ? ? ? ? ?TimeUnit.MILLISECONDS.sleep(20); ? ? ? ? ? ? ? ?new Thread(() -> { MultiValueMap params = new LinkedMultiValueMap<>(); params.add("orderId", "1"); Long result = testRestTemplate.postForObject(url, params, Long.class); if(result != 0) { System.out.println("-------------" + result); } } ).start(); } catch (Exception e) { log.info("error:{}", e.getMessage()); }

} }

@Test public void contextLoads() { }

}


然后開始運行測試代碼,查看一下測試日志和程序日志,均顯示賣了1000后直接顯示SOLD OUT了。分別看一下日志和數據庫:

e8cd70f8-3421-11ed-ba43-dac502259ad0.jpg

e8df77da-3421-11ed-ba43-dac502259ad0.jpg

商品庫存catalog表和訂單明細表sales_order表,都是1000條,沒有問題。

e8f1df10-3421-11ed-ba43-dac502259ad0.jpg

總結:

通過采用分布式鎖和分布式限流,即可實現秒殺流程,當然分布式限流也可以用到很多地方,比如限制某些IP在多久時間訪問接口多少次,都可以的。

令牌桶的限流方式使得請求可以得到更加平滑的處理,不至于瞬間把系統達到最高負載。在這其中其實還有一個小細節,就是Redis的鎖,單機情況下沒有任何問題,如果是集群的話需要注意,一個key被hash到同一個slot的時候沒有問題,如果說擴容或者縮容的話,如果key被hash到不同的slot,程序可能會出問題。

在寫代碼的過程中還出現了一個小問題,就是寫controller的方法的時候,方法一定要聲明成public的,否則自定義的注解用不了,其他service的注解直接變為空,這個問題也是找了很久才找到。

審核編輯:彭靜
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 接口
    +關注

    關注

    33

    文章

    8712

    瀏覽量

    152013
  • 數據庫
    +關注

    關注

    7

    文章

    3852

    瀏覽量

    64724
  • 腳本
    +關注

    關注

    1

    文章

    392

    瀏覽量

    14957

原文標題:用 IDEA 基于SpringBoot2+ mybatis+Redis實現一個秒殺系統(附上源碼)

文章出處:【微信號:AndroidPush,微信公眾號:Android編程精選】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    何氏手機維修秒殺絕殺技術----笫集漏電故障的秒殺技術

    神往,只是我不想傳播,造成何氏秒殺技術是光打雷不下雨。今天不僅要下雨。還是下的特大雨。手機的漏電故障,對于手機維修界難題,就是工作了幾十年的資深工程師往往也是縮手無策。因為有的漏電故障是發熱
    發表于 01-27 23:27

    秒殺到了是德 U1242C

    `昨天秒殺示波器,秒了天。今天終于秒到萬用表。http://mall.jd.com/index-698032.html,2:30的http://mall.jd.com
    發表于 11-11 14:26

    【1元秒殺】萬用表1元購,包郵!數量有限,先到先得~

    一個人開始對電子產品產生興趣時,萬用表可能是他購買的第部測試設備。11月2日,華秋商城貼心為廣大電子工程師、電子發燒友們推出了“萬用表1元秒殺”的福利購活動,還包郵哦~點擊領券(請在手機端上領取
    發表于 11-02 10:57

    如何去實現種基于SpringMVC的電商高并發秒殺系統設計

    參考博客Java高并發秒殺系統API目錄業務場景要解決的問題Redis的使用業務場景首頁倒計時秒殺活動,搶購商品要解決的問題高并發下庫存的控制分布式系統事務處理機制(分布式鎖)
    發表于 01-03 07:50

    嵌入式視頻監控系統的設計與實現

    設計和實現嵌入式視頻監控系統系統由基于嵌入式平臺的視頻服務器、基于PC 機的控制中心和客戶端組成。視頻服務器以嵌入式處理器為硬件平臺
    發表于 08-25 11:51 ?22次下載

    秒殺神券雙組合 vivo 618狂歡眾多炸裂福利勁爆來襲

    的超值福利,將本就白熱化的購機狂歡,推向又一個新的高度。 vivo官網狂歡節超級秒殺日 爆款低至5折限時秒殺,618超級神券限時領取 為進步加大福利放送力度,vivo官網狂歡節超級
    的頭像 發表于 06-13 14:17 ?3033次閱讀

    篇文章秒殺三道區間相關的問題

    經常有讀者問區間相關的問題,今天寫篇文章,秒殺三道區間相關的問題。 所謂區間問題,就是線段問題,讓你合并所有線段、找出線段的交集等等。主要有兩技巧: 1、排序。常見的排序方法就是按照區間起點排序
    的頭像 發表于 10-12 14:54 ?1937次閱讀
    <b class='flag-5'>一</b>篇文章<b class='flag-5'>秒殺</b>三道區間相關的問題

    如何實現微內核操作系統的設計

    設計并實現運行在Bochs虛擬機上的微內核結構的操作系統, 詳細描述了系統中進程管理、進程間通訊、基本內存管理、磁盤服務器以及文件服務
    發表于 11-13 17:28 ?28次下載
    如何<b class='flag-5'>實現</b><b class='flag-5'>一</b><b class='flag-5'>個</b>微內核操作<b class='flag-5'>系統</b>的設計

    阿里的秒殺系統是如何設計的?

    阿里的秒殺系統是怎么設計的?,服務器,redis,調用,后端
    的頭像 發表于 02-20 11:23 ?1997次閱讀
    阿里的<b class='flag-5'>秒殺</b><b class='flag-5'>系統</b>是如何設計的?

    解密高并發業務場景下典型的秒殺系統的架構

    中,就更別提如何構建高并發系統了! 究竟什么樣的系統算是高并發系統?今天,我們就起解密高并發業務場景下典型的秒殺
    的頭像 發表于 11-17 10:32 ?2309次閱讀
    解密高并發業務場景下典型的<b class='flag-5'>秒殺</b><b class='flag-5'>系統</b>的架構

    【源碼版】基于SpringMVC的電商高并發秒殺系統設計思路

    參考博客Java高并發秒殺系統API目錄業務場景要解決的問題Redis的使用業務場景首頁倒計時秒殺活動,搶購商品要解決的問題高并發下庫存的控制分布式系統事務處理機制(分布式鎖)
    發表于 01-12 10:23 ?0次下載
    【源碼版】基于SpringMVC的電商高并發<b class='flag-5'>秒殺</b><b class='flag-5'>系統</b>設計思路

    用Zookeeper怎么實現分布式鎖?

    。但是它僅限于單體項目,也就是說它們只能保證單個JVM應用內線程的順序執行。 如果你部署了多個節點,也就是分布式場景下如何保證不同節點在同時刻只有線程執行呢?場景的業務場景比如秒殺
    的頭像 發表于 05-11 11:02 ?2223次閱讀
    用Zookeeper怎么<b class='flag-5'>實現</b><b class='flag-5'>一</b><b class='flag-5'>個</b>分布式鎖?

    如何控制秒殺商品頁面購買按鈕的點亮

    售空;(4)般是定時上架;(5)時間短、瞬時并發量高; ? 2 秒殺技術挑戰 假設某網站秒殺活動只推出件商品,預計會吸引1萬人參加活動,也就說最大并發請求數是10000,
    的頭像 發表于 06-29 11:12 ?894次閱讀
    如何控制<b class='flag-5'>秒殺</b>商品頁面購買按鈕的點亮

    如何實現malloc

    甚至把malloc當做操作系統所提供的系統調用或C的關鍵字。實際上,malloc只是C的標準庫中提供的普通函數,而且實現malloc的基
    的頭像 發表于 11-13 14:31 ?833次閱讀
    如何<b class='flag-5'>實現</b><b class='flag-5'>一</b><b class='flag-5'>個</b>malloc

    java結合redis秒殺功能

    近年來,隨著電子商務的快速發展,各大電商平臺都推出了各種促銷活動來吸引用戶。秒殺活動作為種高效的促銷方式,能夠在很短的時間內促成大量的交易。然而,高并發場景下的秒殺活動也給系統帶來了
    的頭像 發表于 12-04 11:06 ?649次閱讀
    主站蜘蛛池模板: 日本资源在线 | 精品伊人久久大线蕉地址 | 最新版天堂资源中文官网 | 日韩一级片在线免费观看 | 香蕉久久夜色精品国产2020 | 天天尻 | 三级视频国产 | 四虎国产精品免费观看 | 亚洲区视频在线观看 | 成年网站在线观看 | 四虎影视永久在线精品免费播放 | 深夜网站免费 | 日本高清一本视频 | 亚洲一区二区三区麻豆 | 人人人干 | 欧美黑人性色黄在线视频 | 美女免费视频一区二区三区 | 免费国产成高清人在线视频 | 一区精品视频 | 亚洲一区二区三区免费在线观看 | 日本污污视频 | 手机看片1024欧美 | 2020年亚洲天天爽天天噜 | 五月天免费在线播放 | 成人宗合网 | 亚洲亚洲人成网站在线观看 | 日本污全彩肉肉无遮挡彩色 | 久久福利国产 | 午夜视频在线观看国产www | 亚洲人与牲动交xxxxbbbb | 亚洲综合区图片小说区 | 91大神视频网站 | 国产精品午夜自在在线精品 | 久久午夜宅男免费网站 | 丁香激情综合网 | 一级特黄特黄xxx视频 | 亚洲一区二区中文字幕 | 国产一卡2卡3卡四卡精品网站 | 永久黄网站色视频免费 | 婷婷 夜夜 | 97影院理论午夜论不卡 |