java高级精讲之高并发抢红包~揭开Redis分布式集群与Lua神秘面纱
java高级精讲之高并发抢红包~揭开Redis分布式集群与Lua神秘面纱
Redis企业集群高级应用精品教程【图灵学院】
利用redis + lua解决抢红包高并发的问题
抢红包的需求分析
抢红包的场景有点像秒杀,但是要比秒杀简单点。
因为秒杀通常要和库存相关。而抢红包则可以允许有些红包没有被抢到,因为发红包的人不会有损失,没抢完的钱再退回给发红包的人即可。
另外像小米这样的抢购也要比淘宝的要简单,也是因为像小米这样是一个公司的,如果有少量没有抢到,则下次再抢,人工修复下数据是很简单的事。而像淘宝这么多商品,要是每一个都存在着修复数据的风险,那如果出故障了则很麻烦。
淘宝的专家丁奇有个文章有写到淘宝是如何应对秒杀的:《秒杀场景下MySQL的低效–原因和改进》
http://blog.nosqlfan.com/html/4209.html
基于redis的抢红包方案
下面介绍一种基于redis的抢红包方案。
把原始的红包称为大红包,拆分后的红包称为小红包。
1.小红包预先生成,插到数据库里,红包对应的用户ID是null。生成算法见另一篇blog:http://blog.csdn.net/hengyunabc/article/details/19177877
2.每个大红包对应两个redis队列,一个是未消费红包队列,另一个是已消费红包队列。开始时,把未抢的小红包全放到未消费红包队列里。
未消费红包队列里是json字符串,如{userId:'789', money:'300'}。
3.在redis中用一个map来过滤已抢到红包的用户。
4.抢红包时,先判断用户是否抢过红包,如果没有,则从未消费红包队列中取出一个小红包,再push到另一个已消费队列中,最后把用户ID放入去重的map中。
5.用一个单线程批量把已消费队列里的红包取出来,再批量update红包的用户ID到数据库里。
上面的流程是很清楚的,但是在第4步时,如果是用户快速点了两次,或者开了两个浏览器来抢红包,会不会有可能用户抢到了两个红包?
为了解决这个问题,采用了lua脚本方式,让第4步整个过程是原子性地执行。
下面是在redis上执行的Lua脚本:
- -- 函数:尝试获得红包,如果成功,则返回json字符串,如果不成功,则返回空
- -- 参数:红包队列名, 已消费的队列名,去重的Map名,用户ID
- -- 返回值:nil 或者 json字符串,包含用户ID:userId,红包ID:id,红包金额:money
- -- 如果用户已抢过红包,则返回nil
- if redis.call('hexists', KEYS[3], KEYS[4]) ~= 0 then
- return nil
- else
- -- 先取出一个小红包
- local hongBao = redis.call('rpop', KEYS[1]);
- if hongBao then
- local x = cjson.decode(hongBao);
- -- 加入用户ID信息
- x['userId'] = KEYS[4];
- local re = cjson.encode(x);
- -- 把用户ID放到去重的set里
- redis.call('hset', KEYS[3], KEYS[4], KEYS[4]);
- -- 把红包放到已消费队列里
- redis.call('lpush', KEYS[2], re);
- return re;
- end
- end
- return nil
下面是测试代码:
- public class TestEval {
- static String host = "localhost";
- static int honBaoCount = 1_0_0000;
- static int threadCount = 20;
- static String hongBaoList = "hongBaoList";
- static String hongBaoConsumedList = "hongBaoConsumedList";
- static String hongBaoConsumedMap = "hongBaoConsumedMap";
- static Random random = new Random();
- // -- 函数:尝试获得红包,如果成功,则返回json字符串,如果不成功,则返回空
- // -- 参数:红包队列名, 已消费的队列名,去重的Map名,用户ID
- // -- 返回值:nil 或者 json字符串,包含用户ID:userId,红包ID:id,红包金额:money
- static String tryGetHongBaoScript =
- // "local bConsumed = redis.call('hexists', KEYS[3], KEYS[4]);\n"
- // + "print('bConsumed:' ,bConsumed);\n"
- "if redis.call('hexists', KEYS[3], KEYS[4]) ~= 0 then\n"
- + "return nil\n"
- + "else\n"
- + "local hongBao = redis.call('rpop', KEYS[1]);\n"
- // + "print('hongBao:', hongBao);\n"
- + "if hongBao then\n"
- + "local x = cjson.decode(hongBao);\n"
- + "x['userId'] = KEYS[4];\n"
- + "local re = cjson.encode(x);\n"
- + "redis.call('hset', KEYS[3], KEYS[4], KEYS[4]);\n"
- + "redis.call('lpush', KEYS[2], re);\n"
- + "return re;\n"
- + "end\n"
- + "end\n"
- + "return nil";
- static StopWatch watch = new StopWatch();
- public static void main(String[] args) throws InterruptedException {
- // testEval();
- generateTestData();
- testTryGetHongBao();
- }
- static public void generateTestData() throws InterruptedException {
- Jedis jedis = new Jedis(host);
- jedis.flushAll();
- final CountDownLatch latch = new CountDownLatch(threadCount);
- for(int i = 0; i < threadCount; ++i) {
- final int temp = i;
- Thread thread = new Thread() {
- public void run() {
- Jedis jedis = new Jedis(host);
- int per = honBaoCount/threadCount;
- JSONObject object = new JSONObject();
- for(int j = temp * per; j < (temp+1) * per; j++) {
- object.put("id", j);
- object.put("money", j);
- jedis.lpush(hongBaoList, object.toJSONString());
- }
- latch.countDown();
- }
- };
- thread.start();
- }
- latch.await();
- }
- static public void testTryGetHongBao() throws InterruptedException {
- final CountDownLatch latch = new CountDownLatch(threadCount);
- System.err.println("start:" + System.currentTimeMillis()/1000);
- watch.start();
- for(int i = 0; i < threadCount; ++i) {
- final int temp = i;
- Thread thread = new Thread() {
- public void run() {
- Jedis jedis = new Jedis(host);
- String sha = jedis.scriptLoad(tryGetHongBaoScript);
- int j = honBaoCount/threadCount * temp;
- while(true) {
- Object object = jedis.eval(tryGetHongBaoScript, 4, hongBaoList, hongBaoConsumedList, hongBaoConsumedMap, "" + j);
- j++;
- if (object != null) {
- // System.out.println("get hongBao:" + object);
- }else {
- //已经取完了
- if(jedis.llen(hongBaoList) == 0)
- break;
- }
- }
- latch.countDown();
- }
- };
- thread.start();
- }
- latch.await();
- watch.stop();
- System.err.println("time:" + watch.getTotalTimeSeconds());
- System.err.println("speed:" + honBaoCount/watch.getTotalTimeSeconds());
- System.err.println("end:" + System.currentTimeMillis()/1000);
- }
- }
测试结果20个线程,每秒可以抢2.5万个,足以应付绝大部分的抢红包场景。
如果是真的应付不了,拆分到几个redis集群里,或者改为批量抢红包,也足够应付。
总结:
redis的抢红包方案,虽然在极端情况下(即redis挂掉)会丢失一秒的数据,但是却是一个扩展性很强,足以应付高并发的抢红包方案。
java高级精讲之高并发抢红包~揭开Redis分布式集群与Lua神秘面纱的更多相关文章
- 2017最新技术java高级架构、千万高并发、分布式集群、架构师入门到精通视频教程
* { font-family: "Microsoft YaHei" !important } h1 { color: #FF0 } 15套java架构师.集群.高可用.高可扩展. ...
- RocketMQ核心技术精讲与高并发抗压实战
1:特点 比较吃内存 内存至少1g 默认8g 1:支持集群模型,强调集群无单点,负载均衡以及水平扩展能力2:亿级别的消息堆积能力3:采用零拷贝原理Consumer 消费消息过程,使用了零拷贝 顺序写盘 ...
- Ubuntu-18.04 下使用Nginx搭建高可用,高并发的asp.net core集群
一.实现前的准备 以下是实现简单负载均衡的思路,图中的服务器均为虚拟机 三台Linux服务器,一台用作Nginx负载均衡(192.168.254.139),另外两台用作Asp.Net Core应用程序 ...
- Java Web(1)高并发业务
互联网无时无刻不面对着高并发问题,例如商品秒杀.微信群抢红包.大麦网抢演唱会门票等. 当一个Web系统,在一秒内收到数以万计甚至更多的请求时,系统的优化和稳定是至关重要的. 互联网的开发包括Java后 ...
- JAVA NIO non-blocking模式实现高并发服务器(转)
原文链接:JAVA NIO non-blocking模式实现高并发服务器 Java自1.4以后,加入了新IO特性,NIO. 号称new IO. NIO带来了non-blocking特性. 这篇文章主要 ...
- JAVA NIO non-blocking模式实现高并发服务器
JAVA NIO non-blocking模式实现高并发服务器 分类: JAVA NIO2014-04-14 11:12 1912人阅读 评论(0) 收藏 举报 目录(?)[+] Java自1.4以后 ...
- 【设计模式】Java设计模式精讲之原型模式
简单记录 - 慕课网 Java设计模式精讲 Debug方式+内存分析 & 设计模式之禅-秦小波 文章目录 1.原型模式的定义 原型-定义 原型-类型 2.原型模式的实现 原型模式的通用类图 原 ...
- Java生鲜电商平台-高并发核心技术订单与库存实战
Java生鲜电商平台-高并发核心技术订单与库存实战 一. 问题 一件商品只有100个库存,现在有1000或者更多的用户来购买,每个用户计划同时购买1个到几个不等商品. 如何保证库存在高并发的场景下是安 ...
- Java生鲜电商平台-高并发的设计与架构
Java生鲜电商平台-高并发的设计与架构 说明:源码下载Java开源生鲜电商平台以及高并发的设计与架构文档 对于高并发的场景来说,比如电商类,o2o,门户,等等互联网类的项目,缓存技术是Java项目中 ...
随机推荐
- elasticsearch 支持中英文搜索和混合搜索
环境: ubuntu16.04 安装: elasticsearch 5.22 1. 第一步,安装java apt-get install default-jre apt-get install def ...
- golang ---tcmalloc浅析
总体结构 在tcmalloc内存管理的体系之中,一共有三个层次:ThreadCache.CentralCache.PageHeap,如上图所示.分配内存和释放内存的时候都是按从前到后的顺序,在各个层次 ...
- arcgis server缓存路径修改
由于空间不够用,需要更换瓦片的输出路径,具体的修改方法如下: 1.打开ArcCatalog,打开GIS服务器,找到已经添加的gis服务器,一般都是机器名,如下所示,右键我的gis服务器(admin-t ...
- [Android源码]Android源码之高仿飞鸽传书WIFI热点搜索与创建(一)
(本文详情来源:android源码 http://www.eoeandroid.com/thread-296427-1-1.html 转载请注明出处!) [Android源码分享]飞鸽传书的An ...
- android 推流解决方案
.LocalSocket + MediaRecorder + librtmp
- Nginx-rtmp 直播媒体实时流实现
0. 前言 这段时间在搭建一个IPCamera项目服务器.视频点对点通话,客户端会查看设备端的音视频实时流.为了省流量,是通过P2P进行穿透.但是由于NAT设备的原因和IPV4的枯竭.有些设备是无法进 ...
- 4. OpenAI GPT算法原理解析
1. 语言模型 2. Attention Is All You Need(Transformer)算法原理解析 3. ELMo算法原理解析 4. OpenAI GPT算法原理解析 5. BERT算法原 ...
- 【九天教您南方cass 9.1】 03 编码法绘制地形图
同学们大家好,欢迎收看由老王测量上班记出品的cass9.1视频课程 我是本节课主讲老师九天. 测量空间的[九天教您南方cass]专栏是九天老师专门开设cass免费教学班.希望能帮助那些刚入行的同学,并 ...
- 【css】css 中文字体 unicode 对照表
css 中文字体可以用 unicode 格式来表示,比如“宋体”可以用 \5B8B\4F53 来表示.具体参考下表: 中文名 英文名 unicode 宋体 SimSun \5B8B\4F53 黑体 S ...
- Java如何从服务器获取文件大小?
在Java编程中,如何从服务器获取文件大小? 以下示例演示如何从服务器获取文件大小. package com.yiibai; import java.net.URL; import java.net. ...