实现一个秒杀系统

来源:https://www.cnblogs.com/huangqingshi/p/10325574.html


之前写了如何实现分布式锁和分布式限流,这次我们继续在这块功能上推进,实现一个秒杀系统,采用spring boot 2.x + mybatis+ redis + swagger2 + lombok实现。

先说说基本流程,就是提供一个秒杀接口,然后针对秒杀接口进行限流,限流的方式目前我实现了两种,上次实现的是累计计数方式,这次还有这个功能,并且我增加了令牌桶方式的lua脚本进行限流。

然后不被限流的数据进来之后,加一把分布式锁,获取分布式锁之后就可以对数据库进行操作了。直接操作数据库的方式可以,但是速度会比较慢,咱们直接通过一个初始化接口,将库存数据放到缓存中,然后对缓存中的数据进行操作。

写库的操作采用异步方式,实现的方式就是将操作好的数据放入到队列中,然后由另一个线程对队列进行消费。当然,也可以将数据直接写入mq中,由另一个线程进行消费,这样也更稳妥。

好了,看一下项目的基本结构:

看一下入口controller类,入口类有两个方法,一个是初始化订单的方法,即秒杀开始的时候,秒杀接口才会有效,这个方法可以采用定时任务自动实现也可以。

初始化后就可以调用placeOrder的方法了。在placeOrder上面有个自定义的注解DistriLimitAnno,这个是我在上篇文章写的,用作限流使用。

采用的方式目前有两种,一种是使用计数方式限流,一种方式是令牌桶,上次使用了计数,咱们这次采用令牌桶方式实现。

packagecom.hqs.flashsales.controller;

importcom.hqs.flashsales.annotation.DistriLimitAnno;

importcom.hqs.flashsales.aspect.LimitAspect;

importcom.hqs.flashsales.lock.DistributedLock;

importcom.hqs.flashsales.limit.DistributedLimit;

importcom.hqs.flashsales.service.OrderService;

importlombok.extern.slf4j.Slf4j;

importorg.springframework.beans.factory.annotation.Autowired;

importorg.springframework.data.redis.core.RedisTemplate;

importorg.springframework.data.redis.core.script.RedisScript;

importorg.springframework.stereotype.Controller;

importorg.springframework.web.bind.annotation.GetMapping;

importorg.springframework.web.bind.annotation.PostMapping;

importorg.springframework.web.bind.annotation.ResponseBody;

importjavax.annotation.Resource;

importjava.util.Collections;

/**

*@authorhuangqingshi

*@Date2019-01-23

*/

@Slf4j

@Controller

publicclassFlashSaleController{

@Autowired

OrderService orderService;

@Autowired

DistributedLock distributedLock;

@Autowired

LimitAspect limitAspect;

//注意RedisTemplate用的String,String,后续所有用到的key和value都是String的

@Autowired

RedisTemplate redisTemplate;

privatestaticfinalString LOCK_PRE ="LOCK_ORDER";

@PostMapping("/initCatalog")

@ResponseBody

publicString 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")

publicLongplaceOrder(LongorderId) {

LongsaleOrderId =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);

}

}

returnsaleOrderId;

}

}

令牌桶的方式比直接计数更加平滑,直接计数可能会瞬间达到最高值,令牌桶则把最高峰给削掉了,令牌桶的基本原理就是有一个桶装着令牌,然后又一队人排队领取令牌,领到令牌的人就可以去做做自己想做的事情了,没有领到令牌的人直接就走了(也可以重新排队)。

发令牌是按照一定的速度发放的,所以这样在多人等令牌的时候,很多人是拿不到的。当桶里边的令牌在一定时间内领完后,则没有令牌可领,都直接走了。如果过了一定的时间之后可以再次把令牌桶装满供排队的人领。

基本原理是这样的,看一下脚本简单了解一下,里边有一个key和四个参数,第一个参数是获取一个令牌桶的时间间隔,第二个参数是重新填装令牌的时间(精确到毫秒),第三个是令牌桶的数量限制,第四个是隔多长时间重新填装令牌桶。

-- bucket name

localkey = KEYS[1]

-- token generate interval

localintervalPerPermit =tonumber(ARGV[1])

-- grant timestamp

localrefillTime =tonumber(ARGV[2])

-- limit token count

locallimit =tonumber(ARGV[3])

-- ratelimit time period

localinterval =tonumber(ARGV[4])

localcounter = redis.call('hgetall', key)

iftable.getn(counter) ==0then

-- 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)

return1

elseiftable.getn(counter) ==4then

-- if bucket exists, first we try to refill the token bucket

locallastRefillTime, tokensRemaining =tonumber(counter[2]),tonumber(counter[4])

localcurrentTokens

ifrefillTime > lastRefillTimethen

-- 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.

localintervalSinceLast = refillTime - lastRefillTime

ifintervalSinceLast > intervalthen

currentTokens = limit

redis.call('hset', key,'lastRefillTime', refillTime)

else

localgrantedTokens =math.floor(intervalSinceLast / intervalPerPermit)

ifgrantedTokens >0then

-- ajust lastRefillTime, we want shift left the refill time.

localpadMillis =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)

ifcurrentTokens ==0then

-- we didn't consume any keys

redis.call('hset', key,'tokensRemaining', currentTokens)

return0

else

-- we take 1 token from the bucket

redis.call('hset', key,'tokensRemaining', currentTokens -1)

return1

end

else

error("Size of counter is "..table.getn(counter) ..", Should Be 0 or 4.")

end

看一下调用令牌桶lua的JAVA代码,也比较简单:

publicBooleandistributedRateLimit(Stringkey,Stringlimit,Stringseconds) {

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) {

returnfalse;

}else{

returntrue;

}

}

创建两张简单表,一个库存表,一个是销售订单表:

CREATETABLE`catalog`(

`id`int(11)unsignedNOTNULLAUTO_INCREMENT,

`name`varchar(50)NOTNULLDEFAULT''COMMENT'名称',

`total`int(11)NOTNULLCOMMENT'库存',

`sold`int(11)NOTNULLCOMMENT'已售',

`version`int(11)NULLCOMMENT'乐观锁,版本号',

PRIMARYKEY(`id`)

)ENGINE=InnoDBDEFAULTCHARSET=utf8;

CREATETABLE`sales_order`(

`id`int(11)unsignedNOTNULLAUTO_INCREMENT,

`cid`int(11)NOTNULLCOMMENT'库存ID',

`name`varchar(30)NOTNULLDEFAULT''COMMENT'商品名称',

`create_time`timestampNOTNULLDEFAULTCURRENT_TIMESTAMPONUPDATECURRENT_TIMESTAMPCOMMENT'创建时间',

PRIMARYKEY(`id`)

)ENGINE=InnoDBDEFAULTCHARSET=utf8;

基本已经准备完毕,然后启动程序,打开swagger(http://localhost:8080/swagger-ui.html#),执行初始化方法initCatalog:

日志里边会输出初始化的记录内容,初始化库存为1000:

初始化执行的方法,十分简单,写到缓存中。

@Override

publicvoidinitCatalog()

{

Catalog catalog =newCatalog();

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个线程,然后去进行下单请求:

packagecom.hqs.flashsales;

importlombok.extern.slf4j.Slf4j;

importorg.junit.Test;

importorg.junit.runner.RunWith;

importorg.springframework.beans.factory.annotation.Autowired;

importorg.springframework.boot.test.context.SpringBootTest;

importorg.springframework.boot.test.web.client.TestRestTemplate;

importorg.springframework.test.context.junit4.SpringRunner;

importorg.springframework.util.LinkedMultiValueMap;

importorg.springframework.util.MultiValueMap;

importjava.util.concurrent.TimeUnit;

@Slf4j

@RunWith(SpringRunner.class)

@SpringBootTest(classes = FlashsalesApplication.class, webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)

publicclassFlashSalesApplicationTests{

@Autowired

privateTestRestTemplate testRestTemplate;

@Test

publicvoidflashsaleTest(){

String url ="http://localhost:8080/placeOrder";

for(inti =0; i <3000; i++) {

try{

TimeUnit.MILLISECONDS.sleep(20);

newThread(() -> {

MultiValueMap params =newLinkedMultiValueMap<>();

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

publicvoidcontextLoads(){

}

}

然后开始运行测试代码,查看一下测试日志和程序日志,均显示卖了1000后直接显示SOLD OUT了。分别看一下日志和数据库:

商品库存catalog表和订单明细表sales_order表,都是1000条,没有问题。

总结:

通过采用分布式锁和分布式限流,即可实现秒杀流程,当然分布式限流也可以用到很多地方,比如限制某些IP在多久时间访问接口多少次,都可以的。

令牌桶的限流方式使得请求可以得到更加平滑的处理,不至于瞬间把系统达到最高负载。在这其中其实还有一个小细节,就是Redis的锁,单机情况下没有任何问题,如果是集群的话需要注意,一个key被hash到同一个slot的时候没有问题,如果说扩容或者缩容的话,如果key被hash到不同的slot,程序可能会出问题。

在写代码的过程中还出现了一个小问题,就是写controller的方法的时候,方法一定要声明成public的,否则自定义的注解用不了,其他service的注解直接变为空,这个问题也是找了很久才找到。

代码地址:

https://github.com/stonehqs/flashsales.git

扩展阅读

Redis实现的分布式锁和分布式限流

阿里淘宝双十一秒杀系统设计详解

从构建分布式秒杀系统聊聊限流特技

高并发系统的设计及秒杀实践

细说JDK动态代理的实现原理

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,036评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,046评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,411评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,622评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,661评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,521评论 1 304
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,288评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,200评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,644评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,837评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,953评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,673评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,281评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,889评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,011评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,119评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,901评论 2 355

推荐阅读更多精彩内容