如何应对Redis并发访问带来的问题

news2025/1/19 14:37:48

前言

我们在使用Redis的过程中,难免会遇到并发访问及数据更新的问题。但很多场景对数据的并发修改是很敏感的,比如库存数据如果没有做好并发读取和更新的版本控制,就会导致严重的业务问题。今天就来说说应该如何做好并发访问及数据更新问题。

什么场景需要控制并发访问

需要控制并发访问,说明这些并发的访问可能会对其他的访问造成影响。比如上面提到的库存问题,若同一时期有多个客户端访问商品A的库存数据,并且可能要更更新库存数据,这时候就需要对并发访问进行控制了。

说到底,并发访问需要控制的就是对数据的更新动作。 一般来说,客户端要进行数据更新时可分为2个步骤:

  1. 客户端读取Redis数据到本地。
  2. 确认数据后,修改Redis的数据。

单个访问来看,这个过程并没什么问题。但是并发多了,一分为二的过程就会造成数据错误的问题。这里还是用库存的例子来说:

  • 时间a::客户端1读取到库存=10,我们需要对库存+1=11的操作。
  • 时间b:客户端2读取到的库存也是10,这次要对库存-1=9的操作。
  • 时间c:客户端1将+1后的值11写回到Redis中。
  • 时间d:客户端2将-1后的值9写回到Redis中。

这样下来,很明显能发现库存数据错了。10+1-1 = 10,正确库存是10,而上述场景最后为9。

由此可见,这个一分为二的操作不具有原子性,从而产生了错误的结果。类型这种场景很多,因此我们需要对这些并发访问的场景加以控制。

并发访问的控制方法

Redis并发访问的控制,总的来说有2种方式。分别是加入锁机制让一系列操作原子化

一、加入锁机制

首先第一点,加入锁机制是很常见的解决方案。简单来说就是一个客户端访问数据之前,先要获取锁,等数据操作完之后再解锁。而在这个客户端拥有锁的过程中,其他客户端如果也想访问修改该数据,必须得等锁释放了之后,获取到了锁才行。

加锁这个方案是可以解决并发访问的数据准确问题,但放在redis这个场景中并不是很好。首先,Redis作为缓存本身并发访问就很多,频繁的加锁解锁,会大大降低redis的访问性能;然后,Redis的客户端在要加锁时,需要用到分布式锁。我们又得用额外的精力去维护这个分布式锁。

二、操作原子化

操作原子化,也就是让要执行的一系列动作都保持原子性操作。它的优点就是不需要加入额外的锁机制。并发的数据准确性达到了,对Redis的性能也不会有太大的影响。

Redis要实现原子操作,总结有2种方式:

  • 单命令操作:也就是Redis中的INCR、HINCRBY等命令,直接将简单的加减操作合成一个命令执行;
  • Lua脚本:借助Lua脚本,让多个操作在Lua脚本上实现原子性操作。

1.单命令操作

首先,单命令操作,将数值的加减直接用Redis命令来执行。像string的加减可用INCR、DECR操作,hash列表field的加减可用HINCRBY操作。

比如下面截图,两个客户端在不同时刻读取的linux_pids a值为4,各自+1、-1后a值为4。结果是正确的。

由此可见,用Redis的INCR、DECR等命令可以解决数值简单增减的并发场景。但如果我们对数据的更新不仅仅是简单的加减操作时,Redis的这些命令就无能为力了。此时我们可以考虑另一种方案:Lua脚本。

2.Lua脚本

Lua语言是由C写的,因此支持多平台和系统。从Redis2.6开始,Redis就内置了Lua解释器,我们能直接用Redis客户端来执行lua脚本。

我们可以将需要执行的一系列操作用Lua脚本写好,然后用Redis执行它。Lua脚本的方法能保证原子性操作的原因是:Redis会将Lua脚本一次性执行,也就是说执行Lua脚本是0-1的操作,要么成功,要么失败。可以理解成MySQL的事务特性。

Redis使用lua脚本有2种方式:

  • 客户端中使用:用到script load脚本内容、evalsha等命令执行
  • 直接执行lua脚本。

我们一般用第二种方式来执行。

  1. 客户端使用方法:

先用script load加载脚本命令,再用evalsha执行加载得到的sha1值。

127.0.0.1:6379> script load "return 'hello'"
"1b936e3fe509bcbc9cd0664897bbe8fd0cac101b"
127.0.0.1:6379> evalsha "1b936e3fe509bcbc9cd0664897bbe8fd0cac101b" 0
"hello"
复制代码
  1. 再来看看Redis使用Lua脚本的语法:
redis-cli --eval {lua_path} KEYS[1] KEYS[2]... , ARGV[1] ARGV[2]...

--eval:  		执行lua脚本的命令
{lua_path}: 		lua脚本的路径
KEYS[1] KEYS[2]:       lua脚本中要操作的redis键,我们可以在lua脚本中用KEYS[1],KEYS[2],KEYS[3]指定多个
ARGV[1] ARGV[2]:	传入到lua脚本的参数,在脚本中用ARGV[1],ARGV[2]...来获取。
复制代码

Redis使用lua脚本的场景很多,最经典的案例当属利用lua来控制某个IP的访问频率了。比如说需要防止恶意访问网站的行为,我们规定1分钟内访问次数不能超过30次,实现的方法有很多,比如说漏桶方案、令牌桶方案,但使用最多的还是Redis+lua的分布式限流方案。

我们用lua脚本(test_lua.script)来简单实现一下上述功能,就是1分钟内若访问次数超过30,直接拦截,否则访问次数+1:

-- 限流的key
local limit_key = KEYS[1]
-- 限流次数
local limit_nums = 30
-- 当前访问次数
local current_num = tonumber(redis.call('get', limit_key) or 0)
-- 超出限流次数
if current_num + 1 > limit_num
	then
    return '超出访问次数'
-- 没有超出限流数,访问次数+1
else
  redis.call("INCRBY", limit_key, "1")
  -- 第一次访问,设置过期时间
  if current_num == 0 then
    redis.call("expire", limit_key, "60")
  return current + 1
end
复制代码

用Redis执行,命令如下:

redis-cli --eval test_lua.script limit_key
复制代码

小结

本文介绍了Redis并发访问的控制问题,以及如何保证并发操作的原子化。原子化操作可通过单命令操作和Lua脚本的方式实现。

我们在应对相关问题时,可根据需要选择对应方案解决之。

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

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

相关文章

使用c++实现通讯录管理系统

使用c实现通讯录管理系统 系统中主要实现的功能如下: 1、添加联系人2、显示联系人3、删除联系人4、查找5、修改6、清空7、退出通讯录 添加联系人(姓名、性别、年龄、联系电话、家庭住址)以下步骤; 1、设计联系人结构体2、设计通讯录结构体…

智能穿戴显示屏怎样操作?智能穿戴显示屏具有怎么功能?

随着科技时代的来临,相信每一个人都有一台智能化设备。智能设备不仅在我们日常生活中随处可见,而且智能设备的发展已经渗透到了我们所能看到的每一个角落。智能穿戴技术作为中心是虚拟现实的技术,它不仅实现了便携收发通讯的功能,…

微服务框架 SpringCloud微服务架构 6 Nacos 配置管理 6.3 配置热更新

微服务框架 【SpringCloudRabbitMQDockerRedis搜索分布式,系统详解springcloud微服务技术栈课程|黑马程序员Java微服务】 SpringCloud微服务架构 文章目录微服务框架SpringCloud微服务架构6 Nacos 配置管理6.3 配置热更新6.3.1 配置自动刷新6.3.2 总结6 Nacos 配置…

Linux进阶-进程间通信(ipc)

进程间通信:数据传输、资源共享、事件通知、进程控制。 Linux系统下的ipc 早期unix系统 ipc:管道(数据传输)、信号(事件通知)、fifo(数据传输)。 system-v ipc(贝尔实…

图的拓扑排序(入门篇)

文章目录拓扑排序满足拓扑排序的前提:图中没有环拓扑排序的实现拓扑排序测试拓扑排序 首先要说明一点:拓扑排序是针对图这种数据结构的特有排序。 百度百科对拓扑排序是这样定义的: 上面的解释不是特别好懂,学过离散数学才知道偏…

探索数字化节能降碳 广域铭岛助力电解铝行业碳达峰

近日,工信部等三部门联合印发《有色金属行业碳达峰实施方案》(下称《方案》),要求确保2030年前有色金属行业实现碳达峰。 其中,针对电解铝行业,《方案》提出了优化冶炼产能规模、调整优化产业结构、强化技…

硬件接口和软件接口

文章目录硬件接口IDESCSISATA光纤通道游戏设备RAID卡USBMD设备MP3视频音频软件接口Java里的接口面向对象的接口聊聊软件接口1. 什么是接口2. 诞生3. 早期(1950-1970)4. 快速发展(1970-1990)5. 多元化发展(1990-2010&am…

双胶合透镜初始设计

双胶合透镜是光学系统中不可或缺的基本光学零件之一。对于一个新设计的光学系统,首先根据性能要求对其进行外形尺寸计算,然后就得开始对各光学零部件进行初级像差设计,求解每个零部件的 、C的分配值,最后根据对各个零部件的 、C要…

小白学编程(JS):随机生成验证码

这道例题来自《JavaScipt从入门到精通》(第三版)中的【例6.6】。 书中给出的代码如下&#xff1a; <body><div id"result"> 产生的验证码&#xff1a;</div><input type"button" name"Submit" class"go-wenbenkuan…

基于PHP+MySQL公积金在线办理系统的设计与实现

公积金在线办理系统具有很强的信息指导性特征,采用PHP开发公积金在线办理系统 给web带来了全新的动态效果,具有更加灵活和方便的交互性。让企业、个人更加方便地在网上开展公积金等工作。 住房公积金是国家机关&#xff0c;企事业单位等及其所在在职职工缴存的长期住房储金。住…

RK3588平台开发系列讲解(CAN篇)CAN FD 开发文档

芯片名称内核版本安卓版本RK3588Linux 5.10Android 12🚀返回专栏总目录 文章目录 一、驱动文件二、DTS 节点配置三、内核配置四、CAN FD 通信测试工具五、CAN FD 常用命令接口沉淀、分享、成长,让自己和他人都能有所收获!😄 📢本篇将介绍CAN的使用及调试手段。 一、驱…

ADC的数据读取问题

一、从补码说起 计算机是如何表示负数的呢&#xff1f;这要从补码说起。 在数学中&#xff0c;任意基数的负数都在最前面加上"−"符号&#xff08;负号&#xff09;来表示。 然而&#xff0c;在计算机硬件中&#xff0c;数字都以无符号的二进制形式表示&#xff0…

[附源码]Python计算机毕业设计Django蛋糕购物商城

项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等等。 环境需要 1.运行环境&#xff1a;最好是python3.7.7&#xff0c;…

复旦-华盛顿大学EMBA 二十年20人丨林劲:对自己多一些“标准”

复旦大学-华盛顿大学EMBA20周年校友系列访谈。      一流企业定标准、二流企业做品牌、三流企业做产品。这是被广为传颂的一句话&#xff0c;意思是说要想成为一流企业&#xff0c;就必须成为行业标准的制定者&#xff0c;或至少能够主导标准的制定。尤其当企业需要在国际市…

2022年NPDP新版教材知识集锦--【第四章节】(3)

【实体化设计阶段】(全部获取文末) 实体化设计阶段是从概念定义开始&#xff0c;根据技术和经济性要求&#xff0c;不断进行设计&#xff0c;直至达到可用于制造的详细设计阶段&#xff0c;从而实现可制造性。 3.1联合分析 联合分析(Conjointanalysis)是一种统计分析方法&am…

python中protobuf和json互相转换应用

在实际信息系统开发中&#xff0c;经常会用到各种各样的协议&#xff0c;网络协议常用的有http&#xff0c;tcp&#xff0c;udp等&#xff0c;传输数据格式协议有json&#xff0c;xml&#xff0c;TLV等。本节将给大家介绍一种节省带宽数据协议&#xff0c;谷歌的ProtoBuf协议&a…

使用 Webmin+bind9快速搭建私有DNS服务器

什么是DNS DNS是Domain name system的简称&#xff0c;有些地方也称为Domain name server DNS主要是用于将域名解析为IP地址的协议&#xff0c;有时候也用于将IP地址反向解析成域名&#xff0c;所以DNS可以实现双向解析。 DNS可以使用TCP和UDP的53端口&#xff0c;基本使用U…

HTML篇_二、HTML简介_HTML入门必修第一课

HTML篇_二、HTML简介 一、HTML的基本结构 1.1 HTML的基本结构及解析 基本结构 这里我们先放一段代码块来进行展示&#xff0c;感受一下来自HTML的魅力。然后下文再对这段代码块进行解析。 <!DOCTYPE html> <html><head><meta charset"utf-8&quo…

计算机组成原理习题课第三章-3(唐朔飞)

计算机组成原理习题课第三章-3&#xff08;唐朔飞&#xff09; ✨欢迎关注&#x1f5b1;点赞&#x1f380;收藏⭐留言✒ &#x1f52e;本文由京与旧铺原创&#xff0c;csdn首发&#xff01; &#x1f618;系列专栏&#xff1a;java学习 &#x1f4bb;首发时间&#xff1a;&…

JavaWeb

1、基础概念 静态web&#xff1a;html、css。给人看的数据始终不会发生改变。&#xff08;数据无法持久化&#xff0c;用户无法交互&#xff09; 动态web&#xff1a;①、淘宝、几乎所有网站&#xff1b; ②、给人看的数据始终会发生改变&#xff1b; ③、技术栈Servlet/JSP、…