java基于redis事务的秒杀实现
package com.vian.user.service; import org.junit.Test;
import org.springframework.util.CollectionUtils;
import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig;
import redis.clients.jedis.Transaction; import java.io.IOException;
import java.util.List;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; /** 秒杀测试 */
public class SecondKillTest {
@Test
public void getResult() {
Jedis jedis = JedisPoolUtil.getJedis();
String goodsResult = jedis.get("goodsResult:user102");
System.out.println(goodsResult);
} @Test
public void test() throws IOException, InterruptedException {
/** 初始化商品 */
initGoods(); /** 1000线程抢购100个商品 */
ExecutorService executorService = Executors.newFixedThreadPool(20);
CountDownLatch count = new CountDownLatch(10000);
long startTime = System.currentTimeMillis();
for (int i = 0; i < 10000; i++) {
executorService.execute(new SecondKillHandlder("user" + i));
count.countDown();
}
executorService.shutdown();
count.await();
long time = System.currentTimeMillis() - startTime;
System.out.println("共耗时:" + time + "毫秒");
// JedisPoolUtil.close();
System.in.read();
} /** 初始化商品数量 */
private void initGoods() {
Jedis jedis = JedisPoolUtil.getJedis();
jedis.set("goods:iphone8", "100"); // 设置100个商品
JedisPoolUtil.returnRes(jedis);
} /** 秒杀处理线程 */
private static class SecondKillHandlder implements Runnable {
String goodsKey = "goods:iphone8"; // 监视的key 当前秒杀商品的数量
Jedis jedis;
String userName; public SecondKillHandlder(String userName) {
this.userName = userName;
} @Override
public void run() {
while (true) {
try {
jedis = JedisPoolUtil.getJedis();
// watch 监视一个key,当事务执行之前这个key发生了改变,事务会被打断
jedis.watch(goodsKey);
int currentGoodsCount = Integer.parseInt(jedis.get(goodsKey)); // 当前剩余商品数量
if (currentGoodsCount <= 0) {
System.out.println("商品已抢完," + userName + "---> 抢购失败 XXX");
break;
}
Transaction tran = jedis.multi(); // 开启事务
tran.incrBy(goodsKey, -1); // 商品数量-1
List<Object> exec = tran.exec(); // 执行事务
if (CollectionUtils.isEmpty(exec)) {
System.out.println(userName + "---> 抢购失败,继续抢购");
Thread.sleep(1);
} else {
exec.forEach(
succ -> {
String succStr =
userName
+ "===========================> 抢购到第【"
+ ((100 - currentGoodsCount) + 1)
+ "】份商品,该商品剩余:"
+ succ.toString();
System.out.println(succStr);
jedis.set("goodsResult:" + userName, succStr); // 业务代码,处理抢购成功
});
break;
}
} catch (Exception e) {
e.printStackTrace();
} finally {
if (jedis != null) {
jedis.unwatch();
JedisPoolUtil.returnRes(jedis);
}
}
}
}
} private static class JedisPoolUtil {
private static JedisPool pool; private static void createJedisPool() {
// 建立连接池配置参数
JedisPoolConfig config = new JedisPoolConfig();
// 设置最大连接数
config.setMaxTotal(100);
// 设置最大阻塞时间,记住是毫秒数milliseconds
config.setMaxWaitMillis(1000);
// 设置空间连接
config.setMaxIdle(10);
// 创建连接池
pool = new JedisPool(config, "192.168.31.201", 6379, 2000, null, 3);
} /** 在多线程环境同步初始化 */
private static synchronized void poolInit() {
if (pool == null) createJedisPool();
} /**
* 获取一个jedis 对象
*
* @return
*/
public static Jedis getJedis() { if (pool == null) poolInit();
return pool.getResource();
} /**
* 归还一个连接
*
* @param jedis
*/
public static void returnRes(Jedis jedis) {
pool.returnResource(jedis);
} public static void close() {
pool.close();
}
}
}
java基于redis事务的秒杀实现的更多相关文章
- Java基于redis实现分布式锁(SpringBoot)
前言 分布式锁,其实原理是就是多台机器,去争抢一个资源,谁争抢成功,那么谁就持有了这把锁,然后去执行后续的业务逻辑,执行完毕后,把锁释放掉. 可以通过多种途径实现分布式锁,例如利用数据库(mysql等 ...
- Java基于Redis的分布式锁
分布式锁,其实最终还是要保证锁(数据)的一致性,说到数据一致性,基于ZK,ETCD数据一致性中间件做分数是锁,才是王道.但是Redis也能满足最基本的需求. 参考: https://www.cnblo ...
- 基于redis的cas实现
cas是我们常用的一种解决并发问题的手段,小到CPU指令集,大到分布式存储,都能看到cas的影子.本文假定你已经充分理解一般的cas方案,如果你还不知道cas是什么,请自行百度 我们在进行关系型数据库 ...
- 一天五道Java面试题----第十天(简述Redis事务实现--------->负载均衡算法、类型)
这里是参考B站上的大佬做的面试题笔记.大家也可以去看视频讲解!!! 文章目录 1.简述Redis事务实现 2.redis集群方案 3.redis主从复制的核心原理 4.CAP理论,BASE理论 5.负 ...
- Redis事务和实现秒杀功能的实现
今天带着学生学习了Redis的事务功能,Redis的事务与传统的关系型数据库(如MySQL)有所不同,Redis的事务不能回滚. Redis中使用multi.exec.discard.watch.un ...
- Java实现排行榜基于Redis
访问我的博客 前言 排行榜作为互联网应用中几乎必不可少的一个元素,其能够勾起人类自身对比的欲望,从而来增加商品的销量.排行榜的实现方式基本大同小异,大部分都基于 Redis 的有序集合 sorted ...
- redis watch 加 事务实现秒杀
<?php //redis watch 加 事务实现秒杀 $redis = new redis(); $result = $redis->connect('10.10.10.119 ...
- 基于redis的高并发秒杀的JAVA-DEMO实现!
在Redis的事务中,WATCH命令可用于提供CAS(check-and-set)功能.假设我们通过WATCH命令在事务执行之前监控了多个Keys,倘若在WATCH之后有任何Key的值发生了变化,EX ...
- java事务/springboot事务/redis事务
java事务(数据库事务):jdbc事务--ACID springboot事务:@Transactional--ACID redis事务:命令集合 将redis事务与mysql事务对比: Mysq ...
随机推荐
- DES加密算法应用:分组加密模式
通常,大多数的分组加密算法都是把数据按照64位分组的方式进行加密和解密.但是几乎所有的加密工作所涉及的数据量都远远大于64位,因此就需要不断地重复加密过程,直到处理完所有的分组.这种分组加密中所涉及的 ...
- JAVA EE获取浏览器和操作系统信息
一.原理说明: 1. 浏览器访问服务端时,Http请求头上会带上客户端一些信息,可通过"user-agent"获取. //java获取方法如下,其他语言也有自己获取方法 Stri ...
- SpringCloud微服务Zuul跨域问题
目前项目结构是VUE做前端,后端采用微服务架构,在开发时前端需要跨域请求数据,通过ZuulFilter配置解决了简单跨域请求需要.但当需要在请求的header中增加token信息时,出现了请求失败的情 ...
- jquery-hide//一段hide代码实现异步隐藏
(本篇博客没有什么参考价值,只用于自己未来复习.) 说白了就是通过“父亲”实现异步 代码: <!DOCTYPE html> <html> <head> <sc ...
- 基于mysql的一些sql语法
Sql: distinct: select distinct * from tea; Tea中每行的数据必有不同,若有两行相同的,则只输出一行 Limit: select * from stu lim ...
- 013_针对单个pid的cpu/内存/io的资源占用统计
#!/usr/bin/env python import sys import os import subprocess from decimal import Decimal from decima ...
- 微信、qq可以上网,但是浏览器却不能上网怎么办
问题描述:微信.qq可以上网,但是浏览器却不能上网怎么办? 解决办法(步骤如下):(1)打开360安全卫士,点击更多 (2)进入到更多中,点击断网急救箱 (3)进入到断网急救箱,点击全面诊断 (4)一 ...
- React 系列教程2:编写兰顿蚂蚁演示程序
简介 最早接触兰顿蚂蚁是在做参数化的时候,那时候只感觉好奇,以为是很复杂的东西.因无意中看到生命游戏的 React 实现,所以希望通过兰顿蚂蚁的例子再学习一下 React. 兰顿蚂蚁的规则非常简单: ...
- uniApp——v-for 动态class、动态style
:class="i.themColor" <view v-for="i in htmlJSON" class="column" :c ...
- 转载:用Jquery实现的图片预加载技术,可以实现有序加载和无序加载!
一.背景 我们在做页面的时候,从用户体验的角度出发,肯定是希望用户以最快的速度看到完整的页面信息,但在实际情况中经常会遇到些问题. 比如受网速影响,页面加载素材的时间比较长,页面会出现短时间的错乱或者 ...