Redis-秒杀场景应用
Redis Util实现
package test.jedis; import java.util.List;
import java.util.Set; import redis.clients.jedis.Jedis;
import redis.clients.jedis.Transaction; public class JedisUtil {
public static Jedis redis = new Jedis("localhost", 6379);// 连接redis static{
//初始化购物数据
redis.set("buy:number:1", "10000");
} /**
* 执行事务的过程其他客户端改变了其中的key值,解决数据一致性问题
* 通过watch 对key的监控来实现其他客户端修改数据后,事务取消
* 用法 首先用watch 开始对key的监控 在开启事务,顺序一定要先监控在执行事务
* */
public synchronized static String buy_trans(String userId) throws InterruptedException{
//1、通过对keys 的设计保证 每个用户只能购买一个
if(!redis.exists(userId)){ //2、当数量不购时提示 秒杀完
if(Integer.parseInt(redis.get("buy:number:1"))>0){ //3、redis 事务 监控数量的变化key=buy:number:1 监控1号商品的数量变化
redis.watch("buy:number:1"); //4、开启事务
Transaction tx = redis.multi(); //5、购买用户购买 成功 成功购买
tx.incr(userId); //6、设置数量减1
tx.decr("buy:number:1"); //7、执行事务
List<Object> results = tx.exec(); }else{
return "数量不足";
}
redis.disconnect();
return "抢购购买成功";
}else{
return "该用户已经购买了";
}
} public static Set<String> getKeys(String keys){
return redis.keys(keys);
} }
Controller 端实现
package com.sf.fs.view; import java.util.Set; import javax.servlet.http.HttpServletRequest; import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping; import com.sf.fs.service.SeckillService; /**
* seckill test
* */
@Controller
public class SeckillController { private SeckillService seckillService; public void setSeckillService(SeckillService seckillService) {
this.seckillService = seckillService;
} /**
* 抢购方法
* */
@RequestMapping("/buying")
public String buying(HttpServletRequest req){ //测试用于方便测试用sessionid 作为用户id
String sessionId=req.getSession().getId(); //存储的key 是 user:sessionId
String msg=seckillService.buying("user:"+sessionId);
System.out.println("返回的操作结果:"+msg);
return "succeed";
} /**
* 测试成功抢购的用户数,是否会出现超卖
* */
@RequestMapping("/result")
public void getResult(HttpServletRequest req){
Set<String> sets=seckillService.getUsers("user:*");
System.out.println("成功秒杀到的用户数:"+sets.size());
}
}
Service 实现
package com.sf.fs.service.impl; import java.util.List;
import java.util.Set; import com.sf.fs.service.SeckillService; import test.jedis.JedisUtil; public class SeckillServiceImpl implements SeckillService { public static int number=100; private JedisUtil jedisUtil; public void setJedisUtil(JedisUtil jedisUtil) {
this.jedisUtil = jedisUtil;
} @Override
public String buying(String key) { try {
return jedisUtil.buy_trans(key);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return "操作失败";
} public Set<String> getUsers(String prefixKey){ return jedisUtil.getKeys(prefixKey);
} }
Redis-秒杀场景应用的更多相关文章
- 【转载】秒杀场景下MySQL的低效原因和改进以及Redis的处理
分享的PPT在如下网址: http://www.doc88.com/p-4199037770087.html 秒杀场景下mysql的低效原因和改进 另外有一个篇文章是针对以上内容的总结: http:/ ...
- 重学 Java 设计模式:实战享元模式「基于Redis秒杀,提供活动与库存信息查询场景」
作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言 程序员的上下文是什么? 很多时候一大部分编程开发的人员都只是关注于功能的实现,只 ...
- 秒杀场景下MySQL的低效(转)
秒杀场景下MySQL的低效 2016-01-14 17:12 178人阅读 评论(0) 收藏 举报 最近业务试水电商,接了一个秒杀的活.之前经常看到淘宝的同行们讨论秒杀,讨论电商,这次终于轮到我们自己 ...
- Redis应用场景-转载
1. MySql+Memcached架构的问题 实际MySQL是适合进行海量数据存储的,通过Memcached将热点数据加载到cache,加速访问,很多公司都曾经使用过这样的架构,但随着业务数据量的 ...
- Redis应用场景(转)
(来源:http://www.cnblogs.com/shanyou/archive/2012/09/04/2670972.html) Redis常用数据类型 Redis最为常用的数据类型主要有以下五 ...
- redis(二)Redis适用场景,如何正确的使用
网络IO模型 Memcached是多线程,非阻塞IO复用的网络模型,分为监听主线程和worker子线程,监听线程监听网络连接,接受请求后,将连接描述字pipe 传递给worker线程,进行读写IO, ...
- redis使用场景
Redis应用场景 Redis开创了一种新的数据存储思路,使用Redis,我们不用在面对功能单调的数据库时,把精力放在如何把大象放进冰箱这样的问题上,而是利用Redis灵活多变的数据结构和数据操作 ...
- <转>Redis 应用场景
http://blog.csdn.net/hguisu/article/details/8836819 1. MySql+Memcached 架构的问题 Memcached采用客户端-服务器的架构, ...
- Redis作者谈Redis应用场景
Redis作者谈Redis应用场景 毫无疑问,Redis开创了一种新的数据存储思路,使用Redis,我们不用在面对功能单调的数据库时,把精力放在如何把大象放进冰箱这样的问题上,而是利用Redis灵活多 ...
- Redis应用场景 及其数据对象 string hash list set sortedset
原文地址:http://www.cnblogs.com/shanyou/archive/2012/09/04/2670972.html Redis开创了一种新的数据存储思路,使用Redis,我们不用在 ...
随机推荐
- HealthKit开发快速入门教程之HealthKit数据的操作
HealthKit开发快速入门教程之HealthKit数据的操作 数据的表示 在HealthKit中,数据是最核心的元素.通过分析数据,人们可以看到相关的健康信息.例如,通过统计步数数据,人们可以知道 ...
- Swift语言中为外部参数设置默认值可变参数常量参数变量参数输入输出参数
Swift语言中为外部参数设置默认值可变参数常量参数变量参数输入输出参数 7.4.4 为外部参数设置默认值 开发者也可以对外部参数设置默认值.这时,调用的时候,也可以省略参数传递本文选自Swift1 ...
- mysql之对索引的操作
1. 为什么使用索引? 数据库对象索引与书的目录非常类似,主要是为了提高从表中检索数据的速度.由于数据储存在数据库表中,所以索引是创建在数据库表对象之上的,由表中的一个字段或多个字段生成的键组成,这些 ...
- Win8怎么查看IP地址
win8查看自己IP地址方法一:查看本地网络法 1.首先从桌面右下角的“网络连接图标上”点击右键,然后选择打开网络和共享中心,如下图所示: 打开win8网络和共享中心 2.之后在打开的网络和共享中心窗 ...
- The number of divisors(约数) about Humble Numbers[HDU1492]
The number of divisors(约数) about Humble Numbers Time Limit: 2000/1000 MS (Java/Others) Memory Lim ...
- JavaScript事件流
什么是JS事件流 早期的IE事件传播方向为由上至下,即从document逐级向下传播到目标元素:而Netscape公司的Netscape Navigator则是朝相反的方向传播, 也就是从目标元素开始 ...
- 【POJ】2449 Remmarguts' Date(k短路)
http://poj.org/problem?id=2449 不会.. 百度学习.. 恩. k短路不难理解的. 结合了a_star的思想.每动一次进行一次估价,然后找最小的(此时的最短路)然后累计到k ...
- BZOJ4129: Haruna’s Breakfast
Description Haruna每天都会给提督做早餐! 这天她发现早饭的食材被调皮的 Shimakaze放到了一棵 树上,每个结点都有一样食材,Shimakaze要考验一下她. 每个食材都有一个美 ...
- dpi 、 dip 、分辨率、屏幕尺寸、px、density 关系以及换算(终结版)
首先,说下概念(网上很多帖子几个地方都搞混了,理一下): dip : device independent pixels ,设备无关像素. 我看很多帖子写的五花八门的,关于d的,什么display ...
- Solve error: Cannot open include file: 'X11/Xlocale.h': No such file or directory
When you use FLTK with VS2010, you may get the error: fatal error C1083: Cannot open include file: ' ...