秒殺小案例涨醋,源碼拿走不謝!

背景

雙11逝撬,618等各大電商節(jié)越來越多浴骂,最吸引人的莫不是秒殺商品的活動。那秒殺活動的背后宪潮,又是用什么技術(shù)實現(xiàn)的呢溯警?是不是很好奇趣苏?另外不懂秒殺的,會被面試官虐嗎梯轻?不要好奇食磕,不要擔(dān)心,小編給你介紹一個小案例喳挑,開拓下思維彬伦。

環(huán)境

  1. jdk1.8+
  2. spring boot
  3. redis

思路

  1. 模擬用戶發(fā)起秒殺操作,首先判斷庫存是否足夠伊诵,如果庫存為0单绑,則秒殺結(jié)束

2.如果庫存足夠,則判斷用戶是否拿到該商品的redis分布式鎖日戈,如果拿到鎖询张,則進入下單業(yè)務(wù)邏輯孙乖,如果沒拿到鎖浙炼,則30秒后再次進入秒殺邏輯 3. 進入下單邏輯后,還需要再次判斷下庫存唯袄,庫存為0弯屈,則秒殺結(jié)束,反之生成訂單恋拷。(這里是為了進一步防止超賣發(fā)生)

搭建

生成springboot項目资厉,pom引入相關(guān)依賴

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<dependency>
    <groupId>redis.clients</groupId>
    <artifactId>jedis</artifactId>
</dependency>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
    <groupId>org.projectlombok</groupId>
    <artifactId>lombok</artifactId>
</dependency>

application.yml添加redis相關(guān)配置

各位可以根據(jù)實際情況來配置

server:
  port: 6004
spring:
  application:
    name: spring-boot-ms
# Redis配置
redis:
  host: 你自己的redis的ip地址
  # Redis服務(wù)器連接端口
  port: 你自己的redis的端口
  # Redis服務(wù)器連接密碼(默認(rèn)為空)
  password: liyajie@2021
  timeout: 30000
  # 連接池最大連接數(shù)(使用負(fù)值表示沒有限制)
  maxTotal: 50
  # 連接池中的最大空閑連接
  maxIdle: 10
  numTestsPerEvictionRun: 1024
  timeBetweenEvictionRunsMillis: 30000
  minEvictableIdleTimeMillis: 1800000
  softMinEvictableIdleTimeMillis: 10000
  # 連接池最大阻塞等待時間(使用負(fù)值表示沒有限制)
  maxWaitMillis: 1500
  testOnBorrow: true
  testWhileIdle: true
  blockWhenExhausted: false
  JmxEnabled: true

定義redis配置類

@Configuration
@PropertySource("classpath:application.yml")
public class RedisConfig {
    @Value("${redis.host}")
    private String host;

    @Value("${redis.port}")
    private int port;

    @Value("${redis.timeout}")
    private int timeout;

    @Value("${redis.password}")
    private String password;

    @Value("${redis.maxIdle}")
    private int maxIdle;

    @Value("${redis.maxTotal}")
    private int maxTotal;

    @Value("${redis.maxWaitMillis}")
    private int maxWaitMillis;

    @Value("${redis.blockWhenExhausted}")
    private Boolean blockWhenExhausted;

    @Value("${redis.JmxEnabled}")
    private Boolean JmxEnabled;

    @Bean
    public JedisPool jedisPoolFactory() {
        JedisPoolConfig jedisPoolConfig = new JedisPoolConfig();
        jedisPoolConfig.setMaxIdle(maxIdle);
        jedisPoolConfig.setMaxTotal(maxTotal);
        jedisPoolConfig.setMaxWaitMillis(maxWaitMillis);
        // 連接耗盡時是否阻塞, false報異常,true阻塞直到超時, 默認(rèn)true
        jedisPoolConfig.setBlockWhenExhausted(blockWhenExhausted);
        // 是否啟用pool的jmx管理功能, 默認(rèn)true
        jedisPoolConfig.setJmxEnabled(JmxEnabled);
        jedisPoolConfig.setTestOnBorrow(true);
        JedisPool jedisPool = new JedisPool(jedisPoolConfig, host, port, timeout, password);
        return jedisPool;
    }
}

定義測試controller類

@RestController
@Slf4j
public class MsController {
    @Autowired
    private RedisUtils redisUtils;

    @Autowired
    MsService msService;

    //總庫存
    private long nKuCuen = 0;
    //商品key名字
    private String shangpingKey = "computer_key";
    //獲取鎖的超時時間 秒
    private int timeout = 30 * 1000;

    /**
     * 設(shè)置分布式鎖
     * @param
     * @return boolean
     * @author liyajie
     * @createTime 2021/12/16 14:52
     **/
    @GetMapping("/setnx/{key}/{val}")
    public boolean setnx(@PathVariable String key, @PathVariable String val) {
        return redisUtils.setnx(key, val);
    }

    /**
     * 刪除分布式鎖
     * @param
     * @return int
     * @author liyajie
     * @createTime 2021/12/16 14:52
     **/
    @GetMapping("/delnx/{key}/{val}")
    public int delnx(@PathVariable String key, @PathVariable String val) {
        return redisUtils.delnx(key, val);
    }

    /**
     * 搶單
     * @param
     * @return List<String>
     * @author liyajie
     * @createTime 2021/12/16 14:54
     **/
    @GetMapping("/qiangdan")
    public List<String> qiangdan() {

        //搶到商品的用戶
        List<String> shopUsers = new ArrayList<>();

        //構(gòu)造很多用戶
        List<String> users = Collections.synchronizedList(new ArrayList<String>());
        IntStream.range(0, 100000).parallel().forEach(b -> {
            users.add("神器-" + b);
        });

        //初始化庫存
        nKuCuen = 10;

        //模擬開搶
        users.parallelStream().forEach(b -> {
            String shopUser = this.qiang(b);
            if (!StringUtils.isEmpty(shopUser)) {
                shopUsers.add(shopUser);
            }
        });

        return shopUsers;
    }

    private String qiang(String b) {
        //用戶開搶時間
        long startTime = System.currentTimeMillis();

        //未搶到的情況下,30秒內(nèi)繼續(xù)獲取鎖
        while ((startTime + timeout) >= System.currentTimeMillis()) {
            //商品是否剩余
            if (nKuCuen <= 0) {
                break;
            }
            if (redisUtils.setnx(shangpingKey, b)) {
                //用戶b拿到鎖
                log.info("用戶{}拿到鎖...", b);
                try {
                    //商品是否剩余
                    if (nKuCuen <= 0) {
                        break;
                    }

                    //模擬生成訂單耗時操作蔬顾,方便查看:神器-50 多次獲取鎖記錄
                    try {
                        TimeUnit.SECONDS.sleep(1);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }

                    //搶購成功宴偿,商品遞減,記錄用戶
                    nKuCuen -= 1;

                    //搶單成功跳出
                    log.info("用戶{}搶單成功跳出...所剩庫存:{}", b, nKuCuen);

                    return b + "搶單成功诀豁,所剩庫存:" + nKuCuen;
                } finally {
                    log.info("用戶{}釋放鎖...", b);
                    //釋放鎖
                    redisUtils.delnx(shangpingKey, b);
                }
            } else {
                log.info("用戶{}等待獲取鎖...", b);
                // 用戶b沒拿到鎖窄刘,在超時范圍內(nèi)繼續(xù)請求鎖,不需要處理
                /*if (b.equals("神器-50") || b.equals("神器-69")) {
                    log.info("用戶{}等待獲取鎖...", b);
                }*/
            }
        }
        return "";
    }
}

測試

image.png

總結(jié)

這個秒殺小案例主要用到的知識點就是redis的分布式鎖的唯一性舷胜,希望大家溫故知新娩践。
需要源碼的可以關(guān)注公眾號【溫故知新之java】,更多干活與你分享烹骨。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末翻伺,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子沮焕,更是在濱河造成了極大的恐慌吨岭,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,482評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件峦树,死亡現(xiàn)場離奇詭異辣辫,居然都是意外死亡簿废,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,377評論 2 382
  • 文/潘曉璐 我一進店門络它,熙熙樓的掌柜王于貴愁眉苦臉地迎上來族檬,“玉大人,你說我怎么就攤上這事化戳〉チ希” “怎么了?”我有些...
    開封第一講書人閱讀 152,762評論 0 342
  • 文/不壞的土叔 我叫張陵点楼,是天一觀的道長扫尖。 經(jīng)常有香客問我,道長掠廓,這世上最難降的妖魔是什么换怖? 我笑而不...
    開封第一講書人閱讀 55,273評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮蟀瞧,結(jié)果婚禮上沉颂,老公的妹妹穿的比我還像新娘。我一直安慰自己悦污,他們只是感情好铸屉,可當(dāng)我...
    茶點故事閱讀 64,289評論 5 373
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著切端,像睡著了一般彻坛。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上踏枣,一...
    開封第一講書人閱讀 49,046評論 1 285
  • 那天昌屉,我揣著相機與錄音,去河邊找鬼茵瀑。 笑死间驮,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的瘾婿。 我是一名探鬼主播蜻牢,決...
    沈念sama閱讀 38,351評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼偏陪!你這毒婦竟也來了抢呆?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,988評論 0 259
  • 序言:老撾萬榮一對情侶失蹤笛谦,失蹤者是張志新(化名)和其女友劉穎抱虐,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體饥脑,經(jīng)...
    沈念sama閱讀 43,476評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡恳邀,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,948評論 2 324
  • 正文 我和宋清朗相戀三年懦冰,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片谣沸。...
    茶點故事閱讀 38,064評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡刷钢,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出乳附,到底是詐尸還是另有隱情内地,我是刑警寧澤,帶...
    沈念sama閱讀 33,712評論 4 323
  • 正文 年R本政府宣布赋除,位于F島的核電站阱缓,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏举农。R本人自食惡果不足惜荆针,卻給世界環(huán)境...
    茶點故事閱讀 39,261評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望颁糟。 院中可真熱鬧航背,春花似錦、人聲如沸滚停。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,264評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽键畴。三九已至,卻和暖如春突雪,著一層夾襖步出監(jiān)牢的瞬間起惕,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,486評論 1 262
  • 我被黑心中介騙來泰國打工咏删, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留惹想,地道東北人。 一個月前我還...
    沈念sama閱讀 45,511評論 2 354
  • 正文 我出身青樓督函,卻偏偏與公主長得像嘀粱,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子辰狡,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,802評論 2 345

推薦閱讀更多精彩內(nèi)容