首页 > 数据库 >MySQL for update skip locked 与 for update nowait

MySQL for update skip locked 与 for update nowait

时间:2024-12-25 16:54:20浏览次数:3  
标签:加锁 reserved 队列 skip update 任务 jobs nowait id

理论(下方有实操)

for update skip locked

  • 官方文档:https://dev.mysql.com/doc/refman/8.0/en/innodb-locking-reads.html#innodb-locking-reads-for-update
  • 语法:select语句后跟 for update skip locked
  • 作用:目标对象没有被其它会话加锁则可加锁,被其它会话加了锁就跳过。
  • 解决问题:用来避免锁资源竞争引起的阻塞。
  • 适用场景:
    • 多用于MySQL作为消息队列的存储组件,特别是消费者数量>1的场景,多个worker进程同时检测任务队列中的任务,确保每个任务只被一个worker处理,同时其它worker不会被阻塞(这里的worker进程,可以是不断循环查询队列任务的过程,用毫秒级到分钟级的死循环检测表里是否有要处理的队列任务,加锁是为了避免多个消费者同时处理同一个队列任务引发的并发问题)。
    • 换句话说:加锁的目的是为了避免多个消费者同时消费相同的数据造成重复消费,加skip locked的目的是为了保证其它进程每次扫描到加锁的数据时跳开,避免因排它锁阻塞,保证探测任务快速执行。
  • 坑:
    • 不适用与并发条件下加锁保证数据不出错的场景,因为它遇见目标数据加锁就跳过。(MySQL官方文档原文:Queries that skip locked rows return an inconsistent view of the data. SKIP LOCKED is therefore not suitable for general transactional work. However, it may be used to avoid lock contention when multiple sessions access the same queue-like table.)。
    • 它能跳过仅仅是它能跳过,不代表没加skip locked的X或S锁遇见这个锁不会发生阻塞。
    • MySQL8才有的特性,5.7会报错:You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'skip locked' at line 1。

nowait

  • 官方文档:https://dev.mysql.com/doc/refman/8.0/en/innodb-locking-reads.html#innodb-locking-reads-for-update
  • 语法:select语句后跟 nowait。
  • 作用:目标对象没有被其它会话加锁则可加锁,被其它会话加了锁则返回异常。
  • 解决问题:用来避免锁资源竞争引起的阻塞。
  • 适用场景:类似于skip locked,但应用场景不多,因为MySQL返回的异常,可能会升级为编程语言调用时的报错。
  • 坑:MySQL8才有的特性,5.7会报错:You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'nowait' at line 1。

No BB,Show Code

成品用例

  • 以消息队列表为例,这是一个标准的支持多渠道,多生产者,多消费者队列和延时队列的任务记录表(记录失败的任务表需要另创建,本文略)。
CREATE TABLE `jobs` (
  `id` bigint unsigned NOT NULL AUTO_INCREMENT,
  `queue` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL COMMENT '不同渠道队列执行的优先级,可以是high,default,low等。',
  `payload` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL COMMENT '队列进程所需要的数据,序列化后的json字符串,包含了框架所需要的序列化数据和业务数据。',
  `attempts` tinyint unsigned NOT NULL COMMENT '记录任务已经被尝试执行的次数。每次任务失败后,attempts的值会递增。当attempts达到设定的最大重试次数时,任务将被标记为失败。',
  `reserved_at` int unsigned DEFAULT NULL COMMENT '表示任务被锁定(reserved)的时间戳。当一个worker开始处理任务时,任务会被锁定,以防止其它 worker同时处理相同的任务。reserved_at存储的是记录被锁定时的时间戳。',
  `available_at` int unsigned NOT NULL COMMENT '记录任务应该变为可执行状态的时间戳。将任务推送到队列时,可以选择延迟任务的执行时间。',
  `created_at` int unsigned NOT NULL COMMENT '队列被创建的时间戳。',
  PRIMARY KEY (`id`) USING BTREE,
  KEY `jobs_queue_index` (`queue`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci ROW_FORMAT=DYNAMIC;
  • 添加三条模拟数据,新增了3个队列任务
INSERT INTO `jobs` (`id`, `queue`, `payload`, `attempts`, `reserved_at`, `available_at`, `created_at`) VALUES (1, 'default', '这里是要被处理的任务:xxxxx', 1, NULL, 1735097169, 1735097169);
INSERT INTO `jobs` (`id`, `queue`, `payload`, `attempts`, `reserved_at`, `available_at`, `created_at`) VALUES (2, 'default', '这里是要被处理的任务:xxxxx', 1, NULL, 1735097179, 1735097179);
INSERT INTO `jobs` (`id`, `queue`, `payload`, `attempts`, `reserved_at`, `available_at`, `created_at`) VALUES (3, 'default', '这里是要被处理的任务:xxxxx', 1, NULL, 1735097189, 1735097189);
  • 队列在无任务执行时的扫描任务,SQL是这样的,找到符合条件的数据后更新reserved_at字段为当前时间,并使用worker进程执行队列任务。
select * from `jobs` where `queue` = 'high' and ((`reserved_at` is null and `available_at` <= 1735097879) or (`reserved_at` <= 1735094259)) order by `id` asc limit 1 FOR UPDATE SKIP LOCKED  
select * from `jobs` where `queue` = 'default' and ((`reserved_at` is null and `available_at` <= 1735097879) or (`reserved_at` <= 1735094259)) order by `id` asc limit 1 FOR UPDATE SKIP LOCKED  
select * from `jobs` where `queue` = 'low' and ((`reserved_at` is null and `available_at` <= 1735097879) or (`reserved_at` <= 1735094259)) order by `id` asc limit 1 FOR UPDATE SKIP LOCKED

for update skip locked用法

证明跳过被锁定的行

步骤 会话1 会话2 说明
1 start transaction; start transaction; 双方开启事务
2 select * from jobs where queue = 'default' and ((reserved_at is null and available_at <= 1735097879) or (reserved_at <= 1735094259)) order by id asc limit 1 FOR UPDATE SKIP LOCKED / 会话1返回id为1的数据
3 / select * from jobs where queue = 'default' and ((reserved_at is null and available_at <= 1735097879) or (reserved_at <= 1735094259)) order by id asc limit 1 FOR UPDATE SKIP LOCKED 会话2返回id为2的数据,直接跳过加锁的1
4 update jobs set reserved_at = UNIX_TIMESTAMP() where id = 1; update jobs set reserved_at = UNIX_TIMESTAMP() where id = 2; 将找到的任务标记为队列正在处理
5 commit; commit; 提交事务,结束流程

nowait用法

证明锁定相同范围的数据会报错。

步骤 会话1 会话2 说明
1 start transaction; start transaction; 双方开启事务
2 select * from jobs where id = 3 FOR UPDATE nowait; / 可正常加锁
3 / select * from jobs where id = 3 FOR UPDATE nowait; 报错:3572 - Statement aborted because lock(s) could not be acquired immediately and NOWAIT is set.
4 commit; commit; 提交事务,结束流程

标签:加锁,reserved,队列,skip,update,任务,jobs,nowait,id
From: https://www.cnblogs.com/phpphp/p/18630869

相关文章

  • 【YashanDB知识库】update (子查询) set ORG_ID_STAN -ID 改写
    本文内容来自YashanDB官网,原文内容请见https://www.yashandb.com/newsinfo/7802941.html?templateId=1718516【关键字】update(子查询)setORG_ID_STAN=ID改写【问题描述】update(子查询)setORG_ID_STAN=ID改写【问题原因分析】update(子查询)setORG_ID_STAN=I......
  • Supermicro Update Manager (SUM)工具使用
    1.介绍SupermicroUpdateManager(SUM)是Supermicro提供的一款服务器管理工具,主要用于管理和配置SupermicroX10及以后代主板的BIOS/BMC固件和设置。 2.远程管理1)获取ProductkeyIPMILICENSE值参考:https://blog.csdn.net/a478555/article/details/89449134使用:......
  • 【项目实战】SQL报错注入之updatexml的实现
    SQL报错注入之updatexml的实现updatexml函数通过输入不符合XPATH格式的数据来触发报错,并利用这一点进行SQL注入,通过分析报错信息,可以判断是否存在注入点,并逐步爆出数据库名、表名、字段名以及敏感数据,感兴趣的可以了解一下1.updatexml报错原理updatexml(xml_doument,X......
  • Bug 33121934 - Library cache lock / load lock / mutex x during connection storm
    ConcurrentlogonsforusermaybeaffectedduetoLastSuccessfulLogonTime(LSLT)update:updateuser$setspare6=DECODE(to_char(:2,'YYYY-MM-DD'),'0000-00-00',to_date(NULL),:2)whereuser#=:1(sql_id=9zg9qd9bm4spu)REDISCOVERY......
  • Nessus Professional 10.8 Auto Installer for Windows (updated Dec 2024)
    NessusProfessional10.8AutoInstallerforWindows(updatedDec2024)发布Nessus试用版自动化安装程序,支持macOSSequoia、RHEL9、Ubuntu24.04和Windows请访问原文链接:https://sysin.org/blog/nessus-auto-install-for-windows/查看最新版。原创作品,转载请保......
  • DeprecationWarning: Callback API version 1 is deprecated, update to latest versi
    背景说明        最近在使用MQTT进行发送消息的时候,每次运行都会弹出这个红色的警告,虽然不影响运行,但是看起来怪怪的,于是乎想顺手解决一下,自己使用的mqtt服务器是mosquitto。具体的警告提示如下:问题分析及解决        在Python中使用MQTT时,如果你收到......
  • .NET8升级.NET9,CodeFirst模式迁移Add-Migration执行Update-DataBase报错
    在做netcore开发时,如果net8一直是正常的,只升级了一下框架net9,在使用EntityFrameworkCore的CodeFirst模式进行迁移时,执行Add-Migration后尝试使用Update-DataBase时出现了如下错误。Unhandledexception.System.InvalidOperationException:Anerrorwasgeneratedforwarni......
  • Debiasing Model Updates for Improving Personalized Federated Training为改进个性
    第一部分:解决的问题联邦学习(FL)是一种分布式机器学习方法,允许设备在不共享本地数据的情况下协同训练模型。在个性化联邦学习中,目标是为每个设备训练个性化模型,而不是一个通用的全局模型。然而,由于设备之间数据分布的异质性,传统方法会导致模型偏差。第二部分:解决的方法/idea......
  • Django 的数据库对象object的save() 方法和update()方法
    Django的save()方法默认情况下不会自动进行数据验证。如果需要在保存之前执行数据验证,需要手动调用full_clean()方法,或者通过重写save()方法显式触发验证。update()方法直接执行SQL,不会实例化模型对象,也不会触发模型方法(如save()或full_clean())。用于批量更新数据,性......
  • update一下
    近日自己做的饭......