当前位置: 首页 > news >正文

redis(7)

全局ID生成器: 

全局ID生成器,是一种在分布式系统下用来生成全局唯一ID的工具,一般要满足以下特性

  1. 唯一性
  2. 高可用(随时访问随时生成)
  3. 递增性
  4. 安全性(不能具有规律性)
  5. 高性能(生成ID的速度快)

为了增加ID的安全性,我们不会使用redis自增的数值,而是使用拼接一些其他的信息

就算时间戳相同,也是可以在每一秒内支持2^32个订单

唯一ID组成部分:

符号位:1bit,永远为0,表示此ID永远是正数

时间戳:31bit,以秒为单位,可以使用69年

序列号:32bit,秒内的计数器,可以支持2^32个不同ID

  • Redis自增ID策略
    • 每天一个key,方便统计订单量
    • ID构造是时间戳+计数器
@Component
public class RedisIDWorker {@Autowiredprivate StringRedisTemplate template;public long NextID(String keyPrefix){//1.生成时间戳long current=System.currentTimeMillis();//2.生成序列号//2.1获取到日期的时间,让这一天的订单自增,还可以方便做统计SimpleDateFormat format=new SimpleDateFormat("yyyy:MM:dd");String data=format.format(System.currentTimeMillis());long count= template.opsForValue().increment("increment:"+keyPrefix+":"+data);System.out.println(count+"jjjj");//3.拼接并且返回return current<<32|count;}}

 

@SpringBootTest
class RedisIDWorkerTest {@Autowiredprivate RedisIDWorker worker;@Testvoid nextID() throws InterruptedException {CountDownLatch latch=new CountDownLatch(300);Runnable runnable=new Runnable() {@Overridepublic void run() {Long ID= worker.NextID("order业务");System.out.println(ID);latch.countDown();}};ExecutorService service= Executors.newFixedThreadPool(1);for(int i=0;i<300;i++){service.submit(runnable);}latch.await();System.out.println("线程池中的任务已经全部完成");}
}

实现获取优惠卷:

1)秒杀是否开始或结束,如果尚未开始或已经结束则无法下单

2)库存是否充足,不足则无法下单

加上事务的原因是因为想要进行修改优惠卷的剩余个数的数据库操作和新增订单的操作要么全部执行成功,要么全部执行失败;

@Controller
public class UserController {@Autowiredprivate DemoMapper mapper;@Autowiredprivate RedisIDWorker worker;@RequestMapping("/GetCard")@ResponseBody@Transactionalpublic String GetOrder(Integer cardID,Integer userID){//在这里面还应该加上用户ID对应的用户是否存在//1.进行判断优惠卷ID是否存在Card card=mapper.SelectCardByID(cardID);if(cardID==null){return "当前优惠卷不存在";}//2.判断秒杀是否开始if(new Timestamp(System.currentTimeMillis()).compareTo(card.getStartTime())<0){return "秒杀活动还没有开始";}//2.判断秒杀活动是否结束
//        if(new Timestamp(System.currentTimeMillis()-500000).compareTo(card.getStartTime())>0){
//            return "秒杀活动已经结束";
//        }//3.判断代金卷是否还充足if(card.getCards()<1){return "当前优惠卷已经没有库存了";}//4.进行扣减库存int data= mapper.DecrmentCardData(cardID);//5.进行新创建订单if(data<1){return "获取优惠卷失败,优惠卷已经没有库存了";}Order order=new Order();order.setOrderID((int) worker.NextID("优惠卷秒杀"));//在实际开发中其实可以在session中获取到userID,当前为了实现方便只是在方法中传递了userIDorder.setUserID(userID);order.setCardID(cardID);mapper.InsertOrder(order);return "获取优惠卷成功";}
}

关于超卖(优惠卷被减成负数),超卖问题就是典型的多线程并发安全问题,针对这一问题的常见解决方案就是加锁:

假设线程1再进行查询库存和删除库存的过程中,还没有删除库存,那么有其他线程在中间插入一些逻辑就会造成线程安全问题 

1)线程1进行查询库存发现库存中只剩下一个优惠卷了

2)此时在线程1进行查询库存之后更新库存中的数据之前(cards=cards-1)

3)此时线程2进行尝试获取优惠卷,判断当前库存中只是剩下一个优惠卷了,此时线程2的时间片用完了,进入到休眠操作

4)此时线程1开始进行更新库存数据,此时优惠卷已经为0了

5)但是此时线程2被唤醒,因为唤醒之前已经判断过优惠卷还有1个,感知不到线程1进行更新了库存,此时线程2更新数据库,于是优惠卷就被减为了1

update card set cards=cards-1 where cardID=#{cardID}

 1)悲观锁:

        认为线程安全问题一定会发生,因此在操作数据之前先获取锁,确保线程串行执行

        例如Synchronized、Lock都属于悲观锁
 2)乐观锁:

        认为线程安全不一定会发生,因此不加锁,只是在更新数据时取判断有没有其他线程对数据做了修改

        如果没有修改则认为时安全的,自己才更新数据,一般用于更新数据
        如果已经被其他线程修改说明发生了安全问题,此时可以重试或异常
关于超卖我们可以加一个乐观锁,乐观锁的关键是判断之前查询得到的数据有被修改过,常见的方式有两种:

1)版本号法:顾名思义,就是在修改数据时加入一个version,如果修改的时候version与自己得到的version 不相同,那么就修改失败,可以尝试重试或异常
2)CAS法:就是更改数据,或者删除库存的时候,判断库存是否大于0,如果大于零,则扣除成功

如果说在我们查询库存到真正的修改库存的过程中发现库存数也就是优惠卷的个数没有发生过改变,说明在这段期间没有其他线程来进行插入修改其逻辑,那么就可以执行扣减库存操作

但是如果这样当我们进行模拟100个用户进行并发访问的时候,就会发现:很少的用户能够抢到优惠卷,这是不符合业务逻辑的,但是此时还发现优惠卷的剩余次数还是大于0的,但是用户还抢不到优惠卷,这又是怎么回事呢?

1)假设此时线程1查询库存发现现在库存还有100个优惠卷

2)此时线程2也进行查询库存发现库存还有100个优惠卷

3)此时线程2的时间片用完,进入到阻塞状态

4)此时线程1的用户执行完获取到优惠卷的操作,库存中的优惠卷总数-1

5)此时线程2的用户开始被唤醒,执行获取到优惠卷操作,此时SQL语句执行失败

update card set cards=cards-1 where cardID=#{cardID} and cards=#{count}

因为此时总的库存数已经不和刚才线程2查询出来的库存数相等了,所以线程2执行数据库操作失败,所以线程2的用户获取到优惠卷失败,解决方案还是进行修改SQL语句;

update card set cards=cards-1 where cardID=#{cardID}and cards>0

一人一单的功能:在进行更新优惠卷数目的时候根据优惠卷ID和用户ID来进行查询订单,如果查询的订单不为空,说明之前用户已经下过单了,那么直接返回false

但是此时做多线程并发访问的时候(用户传入了userID和CardID来去访问请求,结果又发现相同的用户ID下了很多单,于是又发生了线程安全问题;


@Controller
public class UserController {@Autowiredprivate DemoMapper mapper;@Autowiredprivate RedisIDWorker worker;@RequestMapping("/GetCard")@ResponseBody@Transactionalpublic String GetOrder(Integer cardID,Integer userID){//在这里面还应该加上用户ID对应的用户是否存在//1.进行判断优惠卷ID是否存在Card card=mapper.SelectCardByID(cardID);if(cardID==null){return "当前优惠卷不存在";}//2.判断秒杀是否开始if(new Timestamp(System.currentTimeMillis()).compareTo(card.getStartTime())<0){return "秒杀活动还没有开始";}//2.判断秒杀活动是否结束
//        if(new Timestamp(System.currentTimeMillis()-500000).compareTo(card.getStartTime())>0){
//            return "秒杀活动已经结束";
//        }//3.判断代金卷是否还充足Order orderDemo=mapper.SelectOrder(userID,cardID);if(orderDemo!=null){return "您当前已经下过单了,无法再次进行购买";}int count=card.getCards();if(count<1){return "当前优惠卷已经没有库存了";}//4.进行扣减库存int data= mapper.DecrmentCardData(cardID);//update card set cards=cards-1 where cardID=#{cardID} and cards=#{count}//在这里面说明如果如果cards的值和上面第三部查询出优惠卷的count值是相等的,那么就直接进行更新操作,否则就执行失败//5.进行新创建订单if(data<1){return "获取优惠卷失败,优惠卷已经没有库存了";}Order order=new Order();order.setOrderID((int) worker.NextID("优惠卷秒杀"));order.setUserID(userID);order.setCardID(cardID);mapper.InsertOrder(order);return "获取优惠卷成功";}
}

这个线程安全问题就很好理解了,假设有100个线程来同时进行访问我们的代码,这100个线程同时并行执行(100个线程同时执行代码的每一句),那么就又会同时插入多个订单了

所以我们要给查询出订单到判断订单到插入订单的这些逻辑进行从加上悲观锁

 此时我们把新增订单和查询订单和修改优惠卷的业务给加锁,但是上面锁的是this锁的是当前对象,但是锁的粒度很大,其实只有当两个userID相同的用户来进行竞争锁就可以了,下面的代码锁定范围就变小了,程序的执行性能也就变高了

但是上面的代码仍然是存在线程安全问题的,由于此方法是添加事务操作的,必须是先释放锁,再去提交事务,但是释放锁并提交事务的这个操作并不是原子的,所以说在释放锁并提交事务的过程中该线程还没有提交订单如果有其他线程(相同的userID)来进行抢优惠卷的操作,那么一定会发生线程安全问题,所以提交的方案应该是该线程提交事务之后再去释放锁

但是上面的代码还是存在着问题,在单体服务项目中,加锁方式是可行的,但是在分布式环境中,这种方式就是不可行的,锁在JVM底层是依靠ObjectMonitor来进行实现的,因为在分布式项目中,有着不同的机器不同的机器存在着不同的JVM,同时不同的机器就有不同的分布式锁,所以要想解决这个问题,只能让多台机器使用同一把锁,多个JVM使用同一把锁

相关文章:

redis(7)

全局ID生成器: 全局ID生成器&#xff0c;是一种在分布式系统下用来生成全局唯一ID的工具&#xff0c;一般要满足以下特性 唯一性高可用(随时访问随时生成)递增性安全性(不能具有规律性)高性能(生成ID的速度快) 为了增加ID的安全性&#xff0c;我们不会使用redis自增的数值&am…...

互联网从业者高频单词 300个

测试 (Test) 软件 (Software) 用例 (Test Case) 缺陷 (Defect) 提交 (Submit) 回归测试 (Regression Testing) 验收测试 (Acceptance Testing) 单元测试 (Unit Testing) 集成测试 (Integration Testing) 性能测试 (Performance Testing) 负载测试 (load Testing) 压…...

初始化vue中data中的数据

当组件的根元素使用了v-if的时候, 并不会初始化data中的数据 如果想完全销毁该组件并且初始化数据,需要在使用该组件的本身添加v-if 或者是手动初始化该组件中的数据 初始化化数据的一些方法 Object.assign(this.$data, this.$options.data()) this.$data&#xff1a;当前的da…...

神经网络的建立-TensorFlow2.x

要学习深度强化学习&#xff0c;就要学会使用神经网络&#xff0c;建立神经网络可以使用TensorFlow和pytorch&#xff0c;今天先学习以TensorFlow建立网络。 直接上代码 import tensorflow as tf# 定义神经网络模型 model tf.keras.models.Sequential([tf.keras.layers.Dense…...

python基于卷积神经网络实现自定义数据集训练与测试

注意&#xff1a; 如何更改图像尺寸在这篇文章中&#xff0c;修改完之后你就可以把你自己的数据集应用到网络。如果你的训练集与测试集也分别为30和5&#xff0c;并且样本类别也为3类&#xff0c;那么你只需要更改图像标签文件地址以及标签内容&#xff08;如下面两图所示&…...

跟着LearnOpenGL学习3--四边形绘制

文章目录 一、前言二、元素缓冲对象三、完整代码四、绘制模式 一、前言 通过跟着LearnOpenGL学习2–三角形绘制一文&#xff0c;我们已经知道了怎么配置渲染管线&#xff0c;来绘制三角形&#xff1b; OpenGL主要处理三角形&#xff0c;当我们需要绘制别的图形时&#xff0c;…...

c#笔记-结构

装箱 结构是值类型。值类型不能继承其他类型&#xff0c;也不能被其他类型继承。 所以它的方法都是确定的&#xff0c;没有虚方法需要在运行时进行动态绑定。 值类型没有对象头&#xff0c;方法调用由编译器直接确定。 但是&#xff0c;如果使用引用类型变量&#xff08;如接…...

Es分布式搜索引擎

目录 一、什么是ES&#xff1f; 二、什么是elk&#xff1f; 三、什么是倒排索引&#xff1f; 四、正向索引和倒排索引的优缺点对比 五、mysql数据库和es的区别&#xff1f; 六、索引库&#xff08;es中的数据库表&#xff09;操作有哪些&#xff1f; 八、ES分片存储原理 …...

open3d 裁剪点云

目录 1. crop_point_cloud 2. crop 3. crop_mesh 1. crop_point_cloud 关键函数 chair vol.crop_point_cloud(pcd) # vol: SelectionPolygonVolume import open3d as o3dif __name__ "__main__":# 1. read pcdprint("Load a ply point cloud, crop it…...

如何对第三方相同请求进行筛选过滤

文章目录 问题背景处理思路注意事项代码实现 问题背景 公司内部多个系统共用一套用户体系库&#xff0c;对外(钉钉)我们是两个客户身份(这里是根据系统来的)&#xff0c;例如当第三方服务向我们发起用户同步请求&#xff1a;是一个更新用户操作&#xff0c;它会同时发送一个 d…...

Go RPC

目录 文章目录 Go RPCHTTP RPCTCP RPCJSON RPC Go RPC Go 标准包中已经提供了对 RPC 的支持&#xff0c;而且支持三个级别的 RPC&#xff1a;TCP、HTTP、JSONRPC。但 Go 的 RPC 包是独一无二的 RPC&#xff0c;它和传统的 RPC 系统不同&#xff0c;它只支持 Go 开发的服务器与…...

真正的智能不仅仅是一个技术问题

智能并不是单一的技术问题&#xff0c;而是一个包括技术、人类智慧、社会制度和文化等多个方面的综合体&#xff0c;常常涉及技术变革、系统演变、运行方式创新、组织适应。智能是指人类的思考、判断、决策和创造等高级认知能力&#xff0c;可以通过技术手段来实现增强和扩展。…...

【数据结构】复杂度包装泛型

目录 1.时间和空间复杂度 1.1时间复杂度 1.2空间复杂度 2.包装类 2.1基本数据类型和对应的包装类 2.2装箱和拆箱 //阿里巴巴面试题 3.泛型 3.1擦除机制 3.2泛型的上界 1.时间和空间复杂度 1.1时间复杂度 定义&#xff1a;一个算法所花费的时间与其语句的执行次数成…...

Ae:绘画面板

Ae菜单&#xff1a;窗口/绘画 Paint 快捷键&#xff1a;Ctrl 8 绘画工具&#xff08;画笔工具、仿制图章工具及橡皮擦工具&#xff09;仅能工作在图层面板上。在使用绘画工具之前&#xff0c;建议先在绘画 Paint面板中查看或进行相关设置。 说明&#xff1a; 如果要在绘画描边…...

常见的锁和zookeeper

zookeeper 本文由 简悦 SimpRead 转码&#xff0c; 原文地址 zhuanlan.zhihu.com 前言 只有光头才能变强。 文本已收录至我的 GitHub 仓库&#xff0c;欢迎 Star&#xff1a;https://github.com/ZhongFuCheng3y/3y 上次写了一篇 什么是消息队列&#xff1f;以后&#xff0c;本来…...

经验总结:(Redis NoSQL数据库快速入门)

一、Nosql概述 为什么使用Nosql 1、单机Mysql时代 90年代,一个网站的访问量一般不会太大&#xff0c;单个数据库完全够用。随着用户增多&#xff0c;网站出现以下问题 数据量增加到一定程度&#xff0c;单机数据库就放不下了数据的索引&#xff08;B Tree&#xff09;,一个机…...

form表单与模板引擎

文章目录 一、form表单的基本使用1、什么是表单2、表单的组成部分3、 <form>标签的属性4、表单的同步提交及缺点&#xff08;1&#xff09; 什么是表单的同步提交&#xff08;2&#xff09; 表单同步提交的缺点&#xff08;3&#xff09; 如何解决表单同步提交的缺点 二、…...

医院检验信息管理系统源码(云LIS系统源码)JQuery、EasyUI

云LIS系统是一种医疗实验室信息管理系统&#xff0c;提供全面的实验室信息管理解决方案。它的主要功能包括样本管理、检测流程管理、报告管理、质量控制、数据分析和仪器管理等。 云LIS源码技术说明&#xff1a; 技术架构&#xff1a;Asp.NET CORE 3.1 MVC SQLserver Redis等…...

React 组件

文章目录 React 组件复合组件 React 组件 本节将讨论如何使用组件使得我们的应用更容易来管理。 接下来我们封装一个输出 “Hello World&#xff01;” 的组件&#xff0c;组件名为 HelloMessage&#xff1a; React 实例 <!DOCTYPE html> <html> <head> &…...

硕士学位论文的几种常见节奏

摘要: 本文描述硕士学位论文的几种目录结构, 特别针对机器学习方向. 1. 基础版 XX算法及其在YY中的应用 针对情况: 只有一篇小论文支撑. 第 1 章: 引言 ( > 5页) 1.1 背景及意义 (应用背景、研究意义, 2 页) 1.2 研究进展及趋势 (算法方面, 2 页) 1.3 论文结构 (1 页) 第 …...

找兄弟单词

描述 定义一个单词的“兄弟单词”为&#xff1a;交换该单词字母顺序&#xff08;注&#xff1a;可以交换任意次&#xff09;&#xff0c;而不添加、删除、修改原有的字母就能生成的单词。 兄弟单词要求和原来的单词不同。例如&#xff1a; ab 和 ba 是兄弟单词。 ab 和 ab 则不…...

python字典翻转教学

目录 第1关 创建大学英语四级单词字典 第2关 合并大学英语四六级词汇字典 第3关 查单词输出中文释义 第4关 删除字典中特定字母开头的单词 第5关 单词英汉记忆训练 第1关 创建大学英语四级单词字典 本关任务&#xff1a;编写一个能创建大学英语四级单词字典的小程序。 测…...

sentinel 随笔 3-降级处理

0. 像喝点东西&#xff0c;但不知道喝什么 先来段源码&#xff0c;看一下 我们在dashboard 录入的降级规则&#xff0c;都映射到哪些字段上 package com.alibaba.csp.sentinel.slots.block.degrade;public class DegradeRule extends AbstractRule {public DegradeRule(String…...

如何解决IP能ping通但无法上网的问题?

当我们在网络环境中遇到无法上网的问题时&#xff0c;可能会尝试使用ping命令来测试网络连接是否正常。如果ping测试成功&#xff0c;说明我们的IP地址能够和网络中其他设备进行通信&#xff0c;但是无法上网。这种情况下&#xff0c;我们需要采取一些措施来解决这个问题。本文…...

Autosar实践-CANTp

文章目录 前言一、CanTp是什么?二、Autosar配置三、诊断数据传输流程1.接收单帧失败,上层没有适当的buffer2.成功接收单帧3.成功发送单帧4.成功接收多帧5.成功发送多帧前言 CANTp模块作为提供数据拆包、组包、流控制传输的服务,在Autosar基础软件通信中起着至关重要的作用。…...

Redis简介

Redis&#xff08;Remote Dictionary Server&#xff09;是一个开源的键值对&#xff08;key-value&#xff09;数据库&#xff0c;支持网络、可基于内存亦可持久化。 Redis的数据结构包括列表&#xff08;List&#xff09;、集合&#xff08;Set&#xff09;、有序集合&#…...

报错问题修改

Vue 项目报错&#xff1a;‘$‘ is not defined ( no-undef ) 错误原因是不认识 $ 符&#xff0c;他是 JQuery 中得符号&#xff0c;引入了 JQuery 文件里的函数报错onclick is not defined问题(作用域问题) window.onload function (){ onload function (){ 第二种方法…...

专访惠众科技|元宇宙应用如何借助3DCAT实时云渲染实现流畅大并发呈现?

当前互联网流量红利已经逐渐消失&#xff0c;营销同质化愈发严重。在这样的背景下&#xff0c;催生了以为元宇宙 焦点的虚拟产业经济。元宇宙在各行各业中以不同形式快速萌生、成长&#xff0c;呈现出多元化的应用场景。尤其是众多品牌&#xff0c;将元宇宙视为品牌建设与营销新…...

加速开放计算产业化,OCTC五大原则瞄准需求痛点

回顾计算产业过去十余载的历程&#xff0c;开放计算始终是一个绕不开的核心焦点。 始于2011年Facebook发起的数据中心硬件开源项目--开放计算项目&#xff08;简称&#xff1a;OCP&#xff09;&#xff0c;开放计算犹如星星之火&#xff0c;不仅迅速形成燎原之势&#xff0c;更…...

【RabbitMQ】安装及六种模式

文章目录 安装rabbitmq镜像访问容器内部15672端口映射到外面的端口地址RabbitMQ六种模式Hello world模式Work queues模式Publish/Subscribe模式交换机fanout类型 Routing模式Topics模式RPC模式 rabbitmq&#xff1a;0->1的学习 学习文档&#xff1a;https://www.cnblogs.com…...