Redis分布式锁解决抢购问题

转:https://segmentfault.com/a/1190000011421467

废话不多说,首先分享一个业务场景-抢购。一个典型的高并发问题,所需的最关键字段就是库存,在高并发的情况下每次都去数据库查询显然是不合适的,因此把库存信息存入Redis中,利用redis的锁机制来控制并发访问,是一个不错的解决方案。

首先是一段业务代码:

@Transactional
public void orderProductMockDiffUser(String productId){//1.查库存int stockNum  = stock.get(productId);if(stocknum == 0){throw new SellException(ProductStatusEnum.STOCK_EMPTY);//这里抛出的异常要是运行时异常,否则无法进行数据回滚,这也是spring中比较基础的   }else{//2.下单orders.put(KeyUtil.genUniqueKey(),productId);//生成随机用户id模拟高并发sotckNum = stockNum-1;try{Thread.sleep(100);} catch (InterruptedExcption e){e.printStackTrace();}stock.put(productId,stockNum);}
}

这里有一种比较简单的解决方案,就是synchronized关键字。

public synchronized void orderProductMockDiffUser(String productId)

这就是java自带的一种锁机制,简单的对函数加锁和释放锁。但问题是这个实在是太慢了,感兴趣的可以可以写个接口用apache ab压测一下。

ab -n 500 -c 100 http://localhost:8080/xxxxxxx

下面就是redis分布式锁的解决方法。首先要了解两个redis指令
SETNX 和 GETSET,可以在redis中文网上找到详细的介绍。
SETNX就是set if not exist的缩写,如果不存在就返回保存value并返回1,如果存在就返回0。
GETSET其实就是两个指令GET和SET,首先会GET到当前key的值并返回,然后在设置当前Key为要设置Value。

首先我们先新建一个RedisLock类:

@Slf4j
@Component
public class RedisService {@Autowiredprivate StringRedisTemplate stringRedisTemplate;/**** 加锁* @param key* @param value 当前时间+超时时间* @return 锁住返回true*/public boolean lock(String key,String value){if(stringRedisTemplate.opsForValue().setIfAbsent(key,value)){//setNX 返回booleanreturn true;}//如果锁超时 ***String currentValue = stringRedisTemplate.opsForValue().get(key);if(!StringUtils.isEmpty(currentValue) && Long.parseLong(currentValue)<System.currentTimeMillis()){//获取上一个锁的时间String oldvalue  = stringRedisTemplate.opsForValue().getAndSet(key,value);if(!StringUtils.isEmpty(oldvalue)&&oldvalue.equals(currentValue)){return true;}}return false;}/**** 解锁* @param key* @param value* @return*/public void unlock(String key,String value){try {String currentValue = stringRedisTemplate.opsForValue().get(key);if(!StringUtils.isEmpty(currentValue)&&currentValue.equals(value)){stringRedisTemplate.opsForValue().getOperations().delete(key);}} catch (Exception e) {log.error("解锁异常");}}
}

这个项目是springboot的项目。首先要加入redis的pom依赖,该类只有两个功能,加锁和解锁,解锁比较简单,就是删除当前key的键值对。我们主要来说一说加锁这个功能。
首先,锁的value值是当前时间加上过期时间的时间戳,Long类型。首先看到用setiFAbsent方法也就是对应的SETNX,在没有线程获得锁的情况下可以直接拿到锁,并返回true也就是加锁,最后没有获得锁的线程会返回false。 最重要的是中间对于锁超时的处理,如果没有这段代码,当秒杀方法发生异常的时候,后续的线程都无法得到锁,也就陷入了一个死锁的情况。我们可以假设CurrentValue为A,并且在执行过程中抛出了异常,这时进入了两个value为B的线程来争夺这个锁,也就是走到了注释*的地方。currentValue==A,这时某一个线程执行到了getAndSet(key,value)函数(某一时刻一定只有一个线程执行这个方法,其他要等待)。这时oldvalue也就是之前的value等于A,在方法执行过后,oldvalue会被设置为当前的value也就是B。这时继续执行,由于oldValue==currentValue所以该线程获取到锁。而另一个线程获取的oldvalue是B,而currentValue是A,所以他就获取不到锁啦。多线程还是有些乱的,需要好好想一想。
接下来就是在业务代码中加锁啦:首要要@Autowired注入刚刚RedisLock类,不要忘记对这个类加一个@Component注解否则无法注入

private static final int TIMEOUT= 10*1000;
@Transactional
public void orderProductMockDiffUser(String productId){long time = System.currentTimeMillions()+TIMEOUT;if(!redislock.lock(productId,String.valueOf(time)){throw new SellException(101,"换个姿势再试试")}//1.查库存int stockNum  = stock.get(productId);if(stocknum == 0){throw new SellException(ProductStatusEnum.STOCK_EMPTY);//这里抛出的异常要是运行时异常,否则无法进行数据回滚,这也是spring中比较基础的   }else{//2.下单orders.put(KeyUtil.genUniqueKey(),productId);//生成随机用户id模拟高并发sotckNum = stockNum-1;try{Thread.sleep(100);} catch (InterruptedExcption e){e.printStackTrace();}stock.put(productId,stockNum);}redisLock.unlock(productId,String.valueOf(time));
}

大功告成了!比synchronized快了不知道多少倍,再也不会被老板骂了!

转载于:https://www.cnblogs.com/duende99/p/11553745.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/461371.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Linux进程编程1——与“进程”相关的常识

以下内容源于网络资源的学习与整理&#xff0c;如有侵权请告知删除。 一、进程的概述 进程&#xff0c;是指一个具有独立功能的程序关于某个数据集合的一次可以并发执行的运行活动&#xff0c;是处于活跃状态的计算机程序&#xff0c;是系统进行资源分配和调度的基本单位。进程…

在成长中遇到的挫折事件对你的影响_多种语言环境中成长的宝宝,会影响说话早晚?其实没有想象的复杂...

关于用多种语言抚养孩子的案例比比皆是&#xff0c;但并不是所有的父母都鼓励这样做&#xff0c;他们被告知这会导致孩子混乱和语言延迟&#xff0c;使他们错过机会之窗。以下是最常见的案例&#xff0c;以及把孩子培养成双语者背后的真实故事。误解一、与多种语言一起长大会使…

java道路级别

第一级&#xff1a;神人&#xff0c;天资过人而又是技术狂热者同时还拥有过人的商业头脑&#xff0c;高瞻远瞩&#xff0c;技术过人&#xff0c;大器也。如丁磊&#xff0c;求伯君。 第二级&#xff1a;高人&#xff0c;有天赋&#xff0c;技术过人但没有过人的商业头脑&#x…

C#求数组中元素的全排列

2019独角兽企业重金招聘Python工程师标准>>> 1.算法描述 全排列的第一项是该数组的升序排列&#xff0c;最后一项是该数组的降序排列。本文中用到的了一个函数FindNextArray&#xff1a;从升序排列开始&#xff0c;不断使用函数FindNextArray&#xff0c;可以遍历全…

PHP+Ajax手机移动端发红包实例

PHPAjax手机移动端发红包实例 基本流程&#xff1a;当输入完红包数量和总金额后&#xff0c;PHP会根据这两个值进行随机分配每个金额&#xff0c;保证每个人都能领取到一个红包&#xff0c;且每个红包金额不等&#xff0c;并且所有红包金额总额等于总金额。 实现原理&#xff1…

Linux进程编程2——与“进程”相关的API

以下内容源于网络资源的学习与整理&#xff0c;如有侵权请告知删除。 一、fork()函数&#xff1a;父进程创建子进程 函数原型 所需头文件 #include <unistd.h> #include <sys/types.h> 函数原型pid_t fork(void);返回值在子进程中返回0&#xff0c;在父进程…

语法和c区别_dockerfile语法

上次写了关于Dockerfile镜像精简之道之后&#xff0c;很多小伙伴找我问&#xff0c;dockerfile语法的一些东西&#xff0c;这边文章就介绍一下dockerfile语法吧FROMFROM是位于我们dockerfile命令的第一层&#xff0c;首先就会使用该命令。当然一个DokcerFile中存在多个镜像时&a…

Dojo学习笔记(三):类化JavaScript

dojo/_base/declare模块是Dojo Toolkit中创建类的基础。declare支持多重继承&#xff0c;这使得开发者能够编写更加灵活的代码并避免代码多次重写。Dojo.Dijit和Dojox模块都使用declare&#xff0c;在这篇文章中&#xff0c;你就知道为什么你也要这样做了。 准备学习 在开始学习…

联想ThinkCentre M8400t-n000等高配电脑重装成xp蓝屏0xc000007b代码

联想ThinkCentre M8400t&#xff0d;n000等高配电脑重装成xp蓝屏0xc000007b代码配置方法&#xff1a;开机进入bios设置界面 &#xff0c;然后把那个硬盘接口从SATA设置成IDE模式。SATA比IDE高级&#xff0c;具有更高的硬盘传输速度&#xff0c;但是早先XP出的时候还不支持SATA&…

Linux进程编程4——父子进程对文件的操作

以下内容源于网络资源的学习与整理&#xff0c;如有侵权请告知删除。 1、子进程继承父进程中打开的文件 &#xff08;1&#xff09;上下文&#xff1a;父进程先用open打开一个文件得到fd&#xff0c;然后再fork创建子进程&#xff0c;之后在父子进程中各自write向fd中写入内容。…

BIOS误删win10引导 delete boot option如何恢复

BIOS误删win10引导 delete boot option如何恢复 其他系统也可以用来参考,解决只是将原来的配置加回去而已 倒了血霉 说个原则,希望大家在发现错误之后第一时间是保护现场不要乱动,如果在百度或别的地方找到了解决方案,先百度一下这个解决方案是否可行,不然不要轻易尝试,这些解决…

fir fpga 不同截止频率_一种新的FIR滤波器系数量化方法

相对于模拟滤波器&#xff0c;数字滤波器具有高精度、高可靠性、可编程改变滤波特性、便于集成等一系列优点&#xff0c;并且理论上可实现近似理想频率特性的滤波性能。经典的数字滤波器主要包括有限脉冲响应(Finite Impulse Response&#xff0c;FIR)滤波器和无限脉冲响应(Inf…

How to: Build a Client Application

转载于&#xff1a;http://msdn.microsoft.com/en-us/library/y6dc64f2(vvs.80).aspx 转载于:https://www.cnblogs.com/aran/archive/2013/02/22/2922438.html

Linux文件和目录权限

前言文件系统权限可以用‘ll’或者‘ls -l’查看第一位表示文件类型&#xff0c;后面9位三三一组分别表示属主&#xff0c;属组&#xff0c;其他用户权限其中r对文件表示可读权限&#xff0c;如cat&#xff0c;tail&#xff0c;more&#xff0c;less等对目录表示可以使用ls命令…

Linux进程编程3——守护进程

以下内容源于网络资源的学习与整理&#xff0c;如有侵权请告知删除。 一、守护进程介绍 1、进程查看命令ps ps -ajx偏向显示各种有关的ID号&#xff1b;ps -aux偏向显示进程各种占用资源 2、&#xff08;用于向进程发送信号的&#xff09;指令kill “kill -信号编号 进程ID”&a…

JSONObject,JSONArray,对象,数组互相转化

json类型对象转化成对象类型 JSONObject.toJavaObject(jsonObj, Object.class)json类型对象转化为List类型 JSONArray.parseArray(JSONObject.toJSONString(object, Object.class))对象转化成String类型的Json数据 JSONObject.toJSONString(object)转载于:https://www.cnblogs.…

pandas用众数填充缺失值_python数据分析包|Pandas-02之缺失值(NA)处理

本篇详解pandas中缺失值&#xff08;Missing data handling&#xff09;处理常用操作。缺失值处理常用于数据分析数据清洗阶段&#xff1b;Pandas中将如下类型定义为缺失值&#xff1a;NaN: ‘’, ‘#N/A’, ‘#N/A N/A’, ‘#NA’, ‘-1.#IND’, ‘-1.#QNAN’,‘-NaN’, ‘-na…

知识点滴:持久层,DAO,API,DAL,BLL,DLL,csproj,sln

知识点滴&#xff1a;持久层&#xff0c;DAO&#xff0c;API&#xff0c;DAL&#xff0c;BLL&#xff0c;DLL&#xff0c;csproj&#xff0c;sln 摘自: http://www.cnblogs.com/niuniu1985/archive/2009/12/10/1620918.html 知识点滴&#xff1a;持久层&#xff0c;DAO&#xf…

系统设计学习

2019独角兽企业重金招聘Python工程师标准>>> 这里原帖地址: http://www.mitbbs.com/article_t/JobHunting/32492515.html 以下为转载内容 我是分割线 稍微总结一下1. 入门级的news feedhttp://www.quora.com/What-are-best-practices-for-building-somethttp://w…

Linux进程编程5——进程间通信(IPC)概述

以下内容源于网络资源的学习与整理&#xff0c;如有侵权请告知删除。 1、为什么需要进程间通信&#xff1f; &#xff08;1&#xff09;进程间通信&#xff08;IPC&#xff09; 指的是2个任意进程之间的通信。&#xff08;2&#xff09;同一个进程在一个地址空间中 同一个进程…