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

    文章

    8595

    浏览量

    151137
  • 数据库
    +关注

    关注

    7

    文章

    3799

    浏览量

    64381
  • 脚本
    +关注

    关注

    1

    文章

    389

    浏览量

    14864

原文标题:用 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 2976次阅读

    篇文章秒杀三道区间相关的问题

    经常有读者问区间相关的问题,今天写篇文章,秒杀三道区间相关的问题。 所谓区间问题,就是线段问题,让你合并所有线段、找出线段的交集等等。主要有两技巧: 1、排序。常见的排序方法就是按照区间起点排序
    的头像 发表于 10-12 14:54 1898次阅读
    <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 1947次阅读
    阿里的<b class='flag-5'>秒杀</b><b class='flag-5'>系统</b>是如何设计的?

    解密高并发业务场景下典型的秒杀系统的架构

    中,就更别提如何构建高并发系统了! 究竟什么样的系统算是高并发系统?今天,我们就起解密高并发业务场景下典型的秒杀
    的头像 发表于 11-17 10:32 2251次阅读
    解密高并发业务场景下典型的<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 2186次阅读
    用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 835次阅读
    如何控制<b class='flag-5'>秒杀</b>商品页面购买按钮的点亮

    如何实现malloc

    甚至把malloc当做操作系统所提供的系统调用或C的关键字。实际上,malloc只是C的标准库中提供的普通函数,而且实现malloc的基
    的头像 发表于 11-13 14:31 783次阅读
    如何<b class='flag-5'>实现</b><b class='flag-5'>一</b><b class='flag-5'>个</b>malloc

    java结合redis秒杀功能

    近年来,随着电子商务的快速发展,各大电商平台都推出了各种促销活动来吸引用户。秒杀活动作为种高效的促销方式,能够在很短的时间内促成大量的交易。然而,高并发场景下的秒杀活动也给系统带来了
    的头像 发表于 12-04 11:06 610次阅读