技术整理:SpringBoot+Redis+lua脚本防止超卖

SpringBoot+redis+lua 防止超卖

一、背景

工作中遇到了有人用 RedisTemplateincrement去做总库存的加减,但是这种方式是保证不了原子性的还是会超卖。

  • redis 是可以保证原子性,但是 RedisTemplate 里面的方法去调用redis是不能保证原子性

二、优化方案

使用 lua 脚本,去执行 加减操作,执行 redis 的命令,来保证原子性

三、重点代码

RedisTemplate 注入

@Configuration
public class RedisConfig {@SuppressWarnings({"rawtypes", "unchecked"})@Bean(name = "redisTemplate")public RedisTemplate<String, Object> redisTemplate(LettuceConnectionFactory lettuceConnectionFactory) {RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>();redisTemplate.setConnectionFactory(lettuceConnectionFactory);Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);ObjectMapper objectMapper = new ObjectMapper();objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);jackson2JsonRedisSerializer.setObjectMapper(objectMapper);redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);redisTemplate.setKeySerializer(new StringRedisSerializer());redisTemplate.afterPropertiesSet();return redisTemplate;}
抢购帮助类并支持集群模式
/*** @Author liyue* @date 2024/3/22 14:31**/
@Component
public class SecKillProvider {private static final String PRODUCT_KEY = "{cluster:}productstock";private static final String SECKILL_SCRIPT = "lua/seckill.lua";@Resourceprivate RedisTemplate<String, Object> redisTemplate;public void initStock(int stock) {//24小时过期RedisUtils.setIfAbsentTimeout(PRODUCT_KEY, stock, 86400);}//+ DateUtil.format(new Date(), DatePattern.NORM_DATE_PATTERN))public boolean seckill(String userId) {//调用lua脚本并执行DefaultRedisScript<Long> redisScript = new DefaultRedisScript<>();redisScript.setResultType(Long.class);//返回类型是Long//lua文件存放在resources目录下的redis文件夹内redisScript.setScriptSource(new ResourceScriptSource(new ClassPathResource(SECKILL_SCRIPT)));Integer result = (Integer) redisTemplate.opsForValue().get(PRODUCT_KEY);System.out.println(result);System.out.println(PRODUCT_KEY);Long stock = redisTemplate.execute(redisScript, Arrays.asList(PRODUCT_KEY));System.out.println("执行完成--=--stock=" + stock);if (stock >= 0) {// 抢购成功,可以继续处理订单等逻辑System.out.println("User " + userId + " seckill success!");return true;} else {// 抢购失败,库存不足System.out.println("User " + userId + " seckill failed!");return false;}}
}
lua 脚本
local stock = tonumber(redis.call('get', KEYS[1]))
if stock and stock > 0 thenredis.call('decr', KEYS[1])return stock - 1
elsereturn -1
end
并发测试类
/*** @Author liyue* @date 2024/3/22 15:14**/
@RunWith(SpringRunner.class)
@SpringBootTest
@Slf4j
public class SecKillProviderTest {@Resourceprivate SecKillProvider secKillProvider;@Testpublic void t() throws InterruptedException {secKillProvider.initStock(10);for (int i = 0; i < 2000; i++) {new Thread(new Mythread(i)).start();}Thread.sleep(10000);}class Mythread implements Runnable {private int num;Mythread(int num) {this.num = num;}@SuppressWarnings("unchecked")SecKillProvider secKillProvider = SpringUtil.getBean("secKillProvider");@Overridepublic void run() {secKillProvider.seckill(String.valueOf(num));}}
}

总结

使用这种方式,读取文件可以优化成sha的方式去去读取。后面再进行调整,测试类可以模拟并发情况。测试没有什么问题。

本文由mdnice多平台发布

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

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

相关文章

数据结构·排序

1. 排序的概念及运用 1.1 排序的概念 排序&#xff1a;排序是将一组“无序”的记录序列&#xff0c;按照某个或某些关键字的大小&#xff0c;递增或递减归零调整为“有序”的记录序列的操作 稳定性&#xff1a;假定在待排序的记录序列中&#xff0c;存在多个具有相同关键字的记…

在MySQL中,如何处理主键冲突的问题?

在MySQL中处理主键冲突的问题时&#xff0c;有几种常用的方法&#xff1a; 1. INSERT IGNORE - 使用 INSERT IGNORE 语句插入数据时&#xff0c;如果主键冲突&#xff08;即主键已经存在于表中&#xff09;&#xff0c;MySQL将忽略此次插入操作&#xff0c;不会更改现有记录&am…

day03_mysql_课后练习 - 参考答案

文章目录 day03_mysql_课后练习mysql练习题第1题第2题第3题第4题第5题 day03_mysql_课后练习 mysql练习题 第1题 案例&#xff1a; 1、创建一个数据库&#xff1a;day03_test01_school 2、创建如下表格 表1 Department表的定义 字段名字段描述数据类型主键外键非空唯一D…

Docker 笔记(七)--打包软件生成镜像

目录 1. 背景2. 参考3. 文档3.1 使用docker container commit命令构建镜像3.1.1 [Docker官方文档-docker container commit](https://docs.docker.com/reference/cli/docker/container/commit/)Description&#xff08;概述&#xff09;Options&#xff08;选项&#xff09;Exa…

【Redis】Redis特性

Redis 认识redisRedis特性在内存中存储数据可编程可扩展性持久化Clustering高可用性 认识redis Redis&#xff0c;英文全称是Remote Dictionary Server&#xff08;远程字典服务&#xff09;&#xff0c;是一个开源的使用ANSIC语言编写、支持网络、可基于内存亦可持久化的日志…

【JAVA】封装与包

。何为封装呢&#xff1f;简单来说 就是套壳屏蔽细节 封装&#xff1a;将数据和操作数据的方法进行有机结合&#xff0c;隐藏对象的属性和实现细节&#xff0c;仅对外公开接口来和对象进行 交互 访问限定符 public&#xff1a;可以理解为一个人的外貌特征&#xff0c;谁都可以…

【webpack】----错误解决【Cannot read properties of undefined (reading ‘tap‘)】

1. 报错场景 安装 webpack-obfuscator 后&#xff0c;进行 js 代码混淆编译的时候报错。 2. 报错截图 3. 错误原因 通常是由于版本不兼容或配置错误引起的。 4. 查询本地 webpack 版本 4.1 查询命令 npm 查询 npm view webpack versionyarn 查询 yarn info webpack ver…

Java学习笔记 | JavaSE基础语法 | 04 | 数组

文章目录 0.前言1.数组2.数组声明2.1 数组定义2.2 数组初始化1.静态初始化2.动态初始化3.区别4.数组的默认初始化值&#xff1a; 2.3 数组名 3.访问数组3.1 索引3.2 访问数组3.3 length属性 4.数组常见问题5.数组内存分析5.1 内存分配5.2 数组内存分配 6.数组的练习练习1&#…

吴恩达深度学习笔记:神经网络的编程基础2.1-2.4

目录 第一门课&#xff1a;神经网络和深度学习 (Neural Networks and Deep Learning)第二周&#xff1a;神经网络的编程基础 (Basics of Neural Network programming)2.1 二分类(Binary Classification)2.2 逻辑回归(Logistic Regression)2.3 逻辑回归的代价函数&#xff08;Lo…

MySQL:数据类型

文章目录 数据类型分类数值类型越界访问bit类型小数类型floatdecimal 字符串类型charvarchar 日期enum和set 数据类型分类 在MySQL数据库中&#xff0c;存在各种各样的数据类型&#xff1a; 针对于上述的这么多类型&#xff0c;本篇就对于这些类型的数据进行一一解释&#xff…

uView Layout 布局

通过基础的 12 分栏&#xff0c;迅速简便地创建布局 注意 如需实现类似宫格的布局&#xff0c;请使用uView的Grid宫格组件&#xff0c;可以设置角标&#xff0c;功能更完善和灵活 #平台差异说明 App&#xff08;vue&#xff09;App&#xff08;nvue&#xff09;H5小程序√√…

STM32--RC522学习记录

一&#xff0c;datasheet阅读记录 1.关于通信格式 2.读寄存器 u8 RC522_ReadReg(u8 address) {u8 addr address;u8 data0x00;addr((addr<<1)&0x7e)|0x80;//将最高位置一表示read&#xff0c;最后一位按照手册建议变为0Spi_Start();//选中从机SPI2_ReadWriteByte(ad…

javaSwing宿舍管理系统(三个角色)

一、 简介 宿舍管理系统是一个针对学校宿舍管理的软件系统&#xff0c;旨在方便学生、宿管和管理员进行宿舍信息管理、学生信息管理以及宿舍评比等操作。该系统使用 Java Swing 进行界面设计&#xff0c;分为三个角色&#xff1a;管理员、宿管和学生。 二、 功能模块 2.1 管…

2024.03.08 校招 实习 内推 面经

绿*泡*泡VX&#xff1a; neituijunsir 交流*裙 &#xff0c;内推/实习/校招汇总表格 1、校招&实习 | 小马智行Pony.ai 2024春季校园招聘启动&可转正实习&#xff08;内推&#xff09; 校招&实习 | 小马智行Pony.ai 2024春季校园招聘启动&可转正实习&#x…

面向中文大模型价值观的评估与对齐研究:“给AI的100瓶毒药”并解毒,守护AI纯净之心

面向中文大模型价值观的评估与对齐研究&#xff1a;“给AI的100瓶毒药”并解毒&#xff0c;守护AI纯净之心 1.简介 随着Large Language Models&#xff08;LLMs&#xff09;的快速发展&#xff0c;越来越多的人开始担心它们可能带来风险。因此&#xff0c;围绕大模型的“安全…

Pytorch 中的forward 函数内部原理

PyTorch中的forward函数是nn.Module类的一部分&#xff0c;它定义了模型的前向传播规则。当你创建一个继承自nn.Module的类时&#xff0c;你实际上是在定义网络的结构。forward函数是这个结构中最关键的部分&#xff0c;因为它指定了数据如何通过网络流动。 单独设计 forward …

Collection与数据结构 数据结构预备知识(一) :集合框架与时间空间复杂度

1.集合框架 1.1 什么是集合框架 Java集合框架,又被称为容器,是定义在java.util包下的一组接口和接口实现的一些类.其主要的表现就是把一些数据放入这些容器中,对数据进行便捷的存储,检索,管理.集合框架底层实现原理其实就是各种数据结构的实现方法,所以在以后的学习中,我们会…

QT(3/22)

1>使用手动连接&#xff0c;将登录框中的取消按钮使用qt4版本的连接到自定义的槽函数中&#xff0c;在自定义的槽函数中调用关闭函数&#xff0c;将登录按钮使用qt5版本的连接到自定义的槽函数中&#xff0c;在槽函数中判断ui界面上输入的账号是否为"admin"&#…

网络编程中的序列化、反序列化与协议

网络编程中的序列化、反序列化与协议 1. 序列化和反序列化的概念2. 序列化、反序列化与协议的关系3. JSON与网络通信 在网络编程中&#xff0c;序列化和反序列化与协议密切相关&#xff0c;它们共同构成了数据在网络中传输的基础。本文将详细介绍序列化、反序列化以及它们与协议…

StarRocks 助力金融营销数字化进化之路

作者&#xff1a;平安银行 数据资产中心数据及 AI 平台团队负责人 廖晓格 平安银行五位一体&#xff0c;做零售金融的领先银行&#xff0c;五位一体是由开放银行、AI 银行、远程银行、线下银行、综合化银行协同构建的数据化、智能化的零售客户经营模式&#xff0c;这套模式以数…