MySQL数据库笔记——版本号机制和CAS(Compare And Swap)

news2025/3/13 19:59:44

大家好,这里是Good Note,关注 公主号:Goodnote,本文详细介绍乐观锁的两种实现方式:版本号机制和CAS(Compare And Swap)。

在这里插入图片描述

文章目录

    • MySQL 内置的并发控制机制
      • MVCC(多版本并发控制)
      • 锁机制
      • 总结
    • 乐观锁实现方式(需要手动实现)
      • 1. 版本号机制
        • 实现步骤
        • 示例
        • 优缺点
      • 2. CAS(Compare And Swap)
        • 实现步骤
        • 示例:Go 中 CAS 实现
        • 优缺点
      • 3. 版本号机制 vs CAS
    • 历史文章

MySQL 内置的并发控制机制

MySQL 内置了强大的并发控制机制,例如 MVCC(多版本并发控制) 和锁机制。这些机制在更高层次上实现了并发控制。自动处理事务隔离和并发冲突,适用于复杂数据库事务管理。

MVCC(多版本并发控制)

  • 核心:基于事务 ID 和 Undo Log,实现高效的读写并发。
  • 特点:快照读无需加锁,写操作使用回滚日志实现隔离性。

锁机制

  1. 按锁粒度表锁行锁(InnoDB 默认)。

  2. 按锁类型共享锁(S 锁)排他锁(X 锁)

  3. 意向锁:用于标记表中是否有行级锁,避免加锁冲突。

  4. 按加锁机制分:悲观锁和乐观锁。

    • 悲观锁:通过显式加锁防止并发冲突。
    • 乐观锁:通过版本号机制和CAS实现,本文重点

锁的详细信息参考:MySQL面试之——锁的分类

总结

机制程序员实现MySQL 支持
版本号机制程序员设计表结构,手动添加 version 字段和条件更新MySQL 不直接支持,由程序员通过 SQL 手动实现
CAS程序员通过 SQL 条件更新模拟类似行为MySQL 不支持硬件级 CAS,仅能通过条件更新实现
MVCC程序员无需手动实现MySQL 内置支持,通过事务 ID 和 Undo Log 实现
  • 程序员实现的版本号机制和 CAS
    • 需要手动设计表结构、编写 SQL。
    • 适用于轻量级业务逻辑的并发控制。

乐观锁实现方式(需要手动实现)

版本号机制CAS(Compare And Swap) 是实现 乐观锁 的两种常见方式,它们的核心思想是通过条件检查来保证并发安全。以下是两种方法的实现详细介绍:


1. 版本号机制

MySQL 本身并未内置对 版本号机制 的直接支持。版本号机制通常由应用程序开发人员在数据库设计和操作层手动实现。

基本原理

  • 为每一条记录添加一个额外的字段(版本号)。
  • 在更新数据时,先检查版本号是否与读取时一致,再执行更新。
  • 版本号的变化表明数据已经被其他事务修改,当前事务需要重新尝试或放弃。
实现步骤
  1. 读取数据和版本号

    • 查询需要更新的记录,并获取当前版本号。
    SELECT version, value FROM table_name WHERE id = 1;
    
  2. 检查版本号并更新

    • 在更新时,检查版本号是否一致。
    • 如果一致,执行更新并将版本号加 1。
    UPDATE table_name
    SET value = 'new_value', version = version + 1
    WHERE id = 1 AND version = 10;
    
  3. 处理并发冲突(如果存在)

    • 如果 WHERE 子句中的 version 不匹配,说明该记录已被其他事务修改,当前事务更新失败。
示例

假设表结构如下:

CREATE TABLE table_name (
    id INT PRIMARY KEY,
    value VARCHAR(255),
    version INT
);

事务 A:

SELECT version, value FROM table_name WHERE id = 1; -- 返回 version = 10
UPDATE table_name SET value = 'new_value', version = version + 1 WHERE id = 1 AND version = 10;

事务 B(同时运行):

SELECT version, value FROM table_name WHERE id = 1; -- 返回 version = 10
UPDATE table_name SET value = 'another_value', version = version + 1 WHERE id = 1 AND version = 10; -- 更新失败
优缺点
  • 优点
    • 不需要加锁,性能高。
    • 简单易实现。
  • 缺点
    • 需要额外的字段存储版本号。
    • 并发冲突时,需要重试或回滚,可能增加系统开销。

2. CAS(Compare And Swap)

MySQL 不直接支持类似硬件级别的 CAS 操作。对于类似 CAS 的功能,依赖程序员通过代码实现【也有说通过 SQL 实现】。

基本原理

  • CAS 是一种原子操作,用于更新某个值时,先比较当前值是否符合预期。
  • 如果当前值符合预期,则执行更新,否则不更新。
实现步骤
  1. 读取数据的当前值

    • 获取目标变量的当前值。
  2. 比较值是否符合预期

    • 如果当前值与预期值一致,说明没有其他线程修改过该值。
  3. 更新数据

    • 在当前值符合预期时,执行更新。
    • 如果值不一致,操作失败,可以选择重试。
示例:Go 中 CAS 实现

Go 中的 CAS 使用 sync/atomic 包提供的 CompareAndSwap 系列方法。

package main

import (
	"fmt"
	"sync/atomic"
)

func main() {
	var counter int32 = 10

	// 期望值是 10,新值是 11
	success := atomic.CompareAndSwapInt32(&counter, 10, 11)

	if success {
		fmt.Printf("Update successful, new value: %d\n", counter)
	} else {
		fmt.Printf("Update failed, current value: %d\n", counter)
	}
}

补充:也有说可以通过SQL实现,SQL实现,本质上和版本号机制一样,不一样的点在于不需要额外字段,直接操作数据值(count)。如下:

-- 假设表结构如下:
CREATE TABLE counter (
    id INT PRIMARY KEY,
    count INT );

-- 当前 count = 10 
UPDATE counter SET count = count + 1 WHERE id = 1 AND count = 10; 
优缺点
  • 优点
    • 无需加锁,性能高。
    • 操作是原子的,由硬件保证一致性。
  • 缺点
    • 存在 ABA 问题(值从 A 改为 B,又改回 A,CAS 不会察觉)。
    • 如果冲突频繁,可能导致多次重试。

3. 版本号机制 vs CAS

特性版本号机制CAS
实现方式基于字段的版本号,依赖 SQL 条件更新或程序逻辑基于硬件支持的原子操作,直接比较并更新
字段要求需要额外的版本号字段(version不需要额外字段,直接操作数据值
适用场景数据库或程序语言中的并发控制程序语言中的并发控制,数据库中可通过条件更新模拟
是否依赖数据库通常依赖 SQL 实现,但也可在程序中实现不依赖数据库,可直接通过程序实现
优点简单易用,适合数据库复杂业务场景高性能,无需锁,硬件原子操作支持
缺点存在重试开销,版本号字段增加存储开销存在 ABA 问题,冲突频繁时重试代价较高

两者适用于不同的场景,但核心思想相同:通过比较条件,确保操作的正确性。

版本号机制和 CAS 的实现方式在不同层面有区别。以下是对它们在 程序员实现MySQL 内部支持 上的区别详细分析:

历史文章

  1. MySQL数据库笔记——数据库三范式
  2. MySQL数据库笔记——存储引擎(InnoDB、MyISAM、MEMORY、ARCHIVE)
  3. MySQL数据库笔记——常见的几种锁分类
  4. MySQL数据库笔记——索引介绍
  5. MySQL数据库笔记——事务介绍
  6. MySQL数据库笔记——索引结构之B+树
  7. MySQL数据库笔记——索引潜规则(回表查询、索引覆盖、索引下推)
  8. MySQL数据库笔记——索引潜规则(最左前缀原则)
  9. MySQL数据库笔记——常见慢查询优化方式
  10. MySQL数据库笔记——日志介绍
  11. MySQL数据库笔记——多版本并发控制MVCC
  12. MySQL数据库笔记——主从复制

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

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

相关文章

使用 commitlint 和 husky 检查提交描述是否符合规范要求

在上一小节中,我们了解了 Git hooks 的概念,那么接下来我们就使用 Git hooks 来去校验我们的提交信息。 要完成这么个目标,那么我们需要使用两个工具: 注意:npm 需要在 7.x 以上版本。 1. commitlint 用于检查提交信…

使用函数求e的近似值(PTA)C语言

自然常数e可以用级数11/1!1/2!⋯1/n!来近似计算。本题要求实现一个计算阶乘的简单函数,使得可以利用该函数,对给定的非负整数n,求该级数的前n1项和。 函数接口定义: double fact( int n ); 其中n是用户传入的参数,函…

使用Clion在ubuntu上进行交叉编译,并在Linux上远程编译五子棋

目录 1.工具以及概念介绍 (1)Clion软件简介 (2)交叉编译 (3)远程编译 2.操作原理 3.详细操作步骤 (1)配置Clion与虚拟机ubuntu的ssh连接 CLion远程开发Ubuntu,并显…

ubuntu如何禁用 Snap 更新

.禁用 Snap 更新(通过修改 snapd 配置) 打开并编辑 /etc/apt/apt.conf.d/50unattended-upgrades文件。 这个文件控制自动更新的行为。 sudo vim /etc/apt/apt.conf.d/50unattended-upgrades 里面有一行将里面的auto改为false即可禁用更新:…

SpringBoot - Spring Profiles 详解

文章目录 Pre官方文档Spring Profiles 详解1. 基本用法2. 激活 Profiles3. 添加 Active Profiles4. Profile Groups5. 在代码中设置 Profiles6. Profile 特定的配置文件 总结 Pre SpringBoot - Spring Boot 中的配置体系Profile全面解读 SpringBoot - spring.profiles.active…

MyBatis如何处理延迟加载?

大家好,我是锋哥。今天分享关于【MyBatis如何处理延迟加载?】面试题。希望对大家有帮助; MyBatis如何处理延迟加载? 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 在 MyBatis 中,延迟加载(Laz…

ELK 使用教程采集系统日志 Elasticsearch、Logstash、Kibana

前言 你知道对于一个系统的上线考察,必备的几样东西是什么吗?其实这也是面试中考察求职者,是否真的做过系统开发和上线的必备问题。包括:服务治理(熔断/限流) (opens new window)、监控 (opens new window)和日志,如果…

重庆市大数据局:基于可信数据空间的公共数据流通利用

在2024年可信数据空间论坛上,重庆市大数据局相关领导发表了题为:基于可信数据空间的公共数据流通利用的演讲,系统介绍了重庆公共数据流通方面的实践经验和案例。 篇幅限制,部分内容如下:

Sublime Text4 4189 安装激活【 2025年1月3日 亲测可用】

-----------------测试时间2025年1月3日------------------- 下载地址 官方网址:https://www.sublimetext.com 更新日志:https://www.sublimetext.com/download V4189 64位:https://www.sublimetext.com/download_thanks?targetwin-x64 ....…

LabVIEW开发中常见硬件通讯接口快速识别

在 LabVIEW 开发中,与硬件进行通讯是实现数据采集与控制的重要环节。准确判断通讯接口类型和协议,可以提高开发效率,减少调试时间。本文结合 LabVIEW 的实际应用,详细介绍如何识别和判断常见硬件通讯接口的定义,并提供…

刷机TP TP-Link-WDR5660【持续更新】

上文中简单介绍了:路由器刷机TP-Link tp-link-WDR5660 路由器升级宽带速度-CSDN博客 步骤如下: 第一步:安装Linux系统 本文使用virtualBox 安装Ubuntu的debian系统,本文不在讲述章 请自行参考:Kali 安装之腾讯云经验…

Linux-Redis哨兵搭建

环境资源准备 主机名IP端口号角色vm1192.168.64.156379/26379mastervm2192.168.64.166379/26379slavevm3192.168.64.176379/26379slave 6379为redis服务暴露端口号、26379为sentinel暴露端口号。 安装Redis # 包文件下载 wget https://github.com/redis/redis/archive/7.2.2…

df.groupby()方法使用表达式分组

# 索引值是否为偶数,分成两组 df.groupby(lambda x:x%20).sum() df.groupby(df.index%20).sum() # 同上这两个写法看似相似,确实都基于索引值来进行分组,但在实现方式上有细微的区别: df.groupby(lambda x: x % 2 0) 这种方式通过…

Python 数据可视化的完整指南

目录 一、为什么选择 Python 进行数据可视化? 二、常用 Python 可视化库及其特点 三、常用图表类型及其代码示例 折线图:用于展示数据随时间或其他连续变量的变化趋势。 柱状图:用于比较不同类别的数据大小。 散点图:用于展示两个变量之间的关系,并发现数据中的模式…

国内Ubuntu环境Docker部署CosyVoice

国内Ubuntu环境Docker部署CosyVoice 本文旨在记录在 国内 CosyVoice项目在 Ubuntu 环境下如何使用 dockermin-conda进行一键部署。 源项目地址: https://github.com/FunAudioLLM/CosyVoice 如果想要使用 dockerpython 进行部署,可以参考我另一篇博客中的…

Git的使用流程(详细教程)

目录 01.Git是什么? 1.1 Git简介 1.2 SVN与Git的最主要的区别 1.3 GIt主要特点 02.Git是干什么的? 2.1.Git概念汇总 2.2 工作区/暂存区/仓库 2.3 Git使用流程 03.Git的安装配置 3.1 Git的配置文件 3.2 配置-初始化用户 3.3 Git可视化…

ImageNet 2.0?自动驾驶数据集迎来自动标注新时代

引言: 3DGS因其渲染速度快和高质量的新视角合成而备受关注。一些研究人员尝试将3DGS应用于驾驶场景的重建。然而,这些方法通常依赖于多种数据类型,如深度图、3D框和移动物体的轨迹。此外,合成图像缺乏标注也限制了其在下游任务中的…

npm install --global windows-build-tools --save 失败

注意以下点 为啥下载windows-build-tools,是因为node-sass4.14.1 一直下载不成功,提示python2 没有安装,最终要安装这个,但是安装这个又失败,主要有以下几个要注意的 1、node 版本 14.21.3 不能太高 2、管理员运行 …

Beamer-LaTeX学习(教程批注版)【1】

该文档总体由beamer-latex的教程而来,由耳东小白以自身学习路径整理。因其中要点基本按照教程的顺序和结构整理,故而不能称之为完全原创,但也不是翻译,更不是抄袭,是个人自学笔记和批注,其中添加了小白个人…

wx005基于springboot+vue+uniapp的大学生心理健康测评管理系统小程序

开发语言:Java框架:springbootuniappJDK版本:JDK1.8服务器:tomcat7数据库:mysql 5.7(一定要5.7版本)数据库工具:Navicat11开发软件:eclipse/myeclipse/ideaMaven包&#…