mysql 更新时,旧值与新值相同会怎么做?

文章目录

  • 1 问题描述
  • 2 验证
    • 2.1 验证猜想1
    • 2.2 验证猜想2
  • 3 结论
  • 4 mysql 为什么这么设计呢?

1 问题描述

创建一张表t,插入一行数据

mysql> CREATE TABLE `t` (
`id` int(11) NOT NULL primary key auto_increment,
`a` int(11) DEFAULT NULL
) ENGINE=InnoDB;
insert into t values(1,2);

现在表t中已经有id = 1,a=2这一行
在这里插入图片描述
执行更新语句,更新id=1这一样,将a的值修改为2(逻辑上值不变)

 update t set a = 2 where id = 1;

在这里插入图片描述
可以看到返回结果,0 rows affected。
仅从现象上看,mysql在处理这个条sql语句时,可能是以下三种情况:

  1. 命令执行到server层返回,因为执行更新之前需要先读出数据,server层的执行器拿到id=1的这一行,发现a=2,于是不会调用InnoDB修改行字段的接口,直接返回。没加写锁
  2. 命令执行到Innodb层返回,server层调用了InnoDB修改行字段的接口,InnoDB对数据加上了写锁,然后进行当前读,读出最新的a的值是2,发现与旧值相同,于是不更新,直接返回。加了写锁,没执行更新
  3. InnoDB认真执行了修改操作,InnoDB对数据加上了写锁,然后进行当前读,读出最新的a的值是2,发现与旧值相同,但是还是执行了一次更新操作,将2赋值给a,然后记录到redo log中,经过两阶段提交,结束事务。也就是一个完整的,正常的更新过程

2 验证

2.1 验证猜想1

猜想1很好验证,如果没加写锁,让事务A和事务B并发的执行更新语句,如果后执行的事务没有被阻塞,则证明猜想1正确。

时间事务A事务B
t0begin;
t1begin;
t2update t set a = 2 where id = 1;
t3update t set a = 2 where id = 1;
t4commit;
t5commit;

事务A
在这里插入图片描述

事务B
在这里插入图片描述
事务B的update语句被阻塞,等到事务A commit后才会提交。说明update过程中是加锁了的,猜想1不正确。

2.2 验证猜想2

猜想2是加锁,但是没有执行更新逻辑。可以采用InnoDB,MVCC机制来验证,MVCC就是在事务整个过程中会拿到一致性视图,对普通读进行复用,达到可重复读的隔离级别。

时间事务A事务B
t0begin;
t1select * from t where id = 1 // (1,2)
t2begin;
t3update t set a = 3 where id = 1;
t4commit;
t5select * from t where id = 1 // (1,2)
t6commit;

由于MVCC的机制,且InnooDB引擎默认设置的隔离级别是可重复度,所以事务A在开始的时候就会创建一致性视图,整个事务过程中进行复用。
所以,第一句select和第二句select查到的都是(1,2),因为事务B的更新对事务A是不可见的。
接下来,我们对事务A再加一条更新的sql语句:

时间事务A事务B
t0begin;
t1select * from t where id = 1 // (1,2)
t2begin;
t3update t set a = 3 where id = 1;
t4commit;
t5update t set a = 3 where id = 1;
t6select * from t where id = 1 // (1,3)
commit;

t5时刻,事务A执行了更新,更新语句是当前读,不是快照读,所以更新语句会读取到当前表中最新的数据,事务A读出来此时a=3,而自己要更新的也是3,旧值与要更新的新值相同,根据猜想2,事务A此次更新不会执行。也就是,InnoDB 的undo log中保留的行数据还是事务B更新的版本,而这个版本的更新是对事务A是不可见的,也就是后续的select语句读出来的结果应该还是(1,2)。但此时,事务A读出来的结果却是(1,3)。说明猜想2错误,InnoDB确实做了更新,并记录了undo log 和 redo log。

3 结论

mysql要更新一条行记录是,即使这条行记录的旧值与要更新的新值相同,mysql也会按部就班地进行更新。

4 mysql 为什么这么设计呢?

我想有以下几点原因:

  1. 数据一致性,InnoDB会按照用户要求的进行操作,即使这个操作显而易见地可以优化,比如要更新的值与旧值相同。这是为了保证数据一致性,如果让InnoDB决定哪些操作需要跳过,可能会导致数据不一致。
  2. 事务,因为update这条语句可能只是这个事务的其中一环,如果这个语句不按照正常的更新流程走,再发生事务回滚时,就回滚不了。
  3. 触发器,如果有表触发器,比如只要update执行了就触发的触发器,如果innodb自行跳过,这些触发器就不会执行。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

9.登入页面

登入页面 在pages中新建页面login 修改代码 <template><view></view> </template><script setup></script><style lang"scss"></style>添加头像组件 官网 https://vkuviewdoc.fsq.pub/components/avatar.html …

vue:功能【xlsx】动态行内合并

场景&#xff1a;纯前端导出excel数据&#xff0c;涉及到列合并、行合并。 注&#xff09;当前数据表头固定&#xff0c;行内数据不固定。以第一列WM为判断条件&#xff0c;相同名字的那几行数据合并单元格。合并的那几行数据&#xff0c;后面的列按需求进行合并。 注&#x…

github 如何关闭 2FA

一开始按照各种教程都找不到&#xff0c;新版的太小了&#xff0c; https://github.com/settings/security

HTML实现卷轴动画完整源码附注释

动画效果截图 页面的html结构代码 <!DOCTYPE html> <html> <head lang=

【Maven入门篇】(3)依赖配置,依赖传递,依赖范围,生命周期

&#x1f38a;专栏【Maven入门篇】 &#xfeff;> &#x1f354;喜欢的诗句&#xff1a;更喜岷山千里雪 三军过后尽开颜。 &#xfeff;> &#x1f386;音乐分享【The truth that you leave】 &#xfeff;> &#x1f970;欢迎并且感谢大家指出我的问题 文章目录 &…

(四)Android布局类型(线性布局LinearLayout)

线性布局&#xff08;LinearLayout&#xff09;&#xff1a;按照一定的方向排列组件&#xff0c;方向主要分为水平方向和垂直方向。方向的设置通过属性android:orientation设置 android:orientation 其取值有两种 水平方向&#xff1a;android:orientation"horizontal&…

【精品】递归查询数据库 获取树形结构数据 通用方法

数据库表结构 实体类基类 Getter Setter ToString public class RecursionBean {/*** 编号*/private Long id;/*** 父权限ID&#xff0c;根节点的父权限为空*/JsonIgnoreprivate Long pid;private List<? extends RecursionBean> children;/*** 递归查询子节点** param…

申请双软认证需要哪些材料?软件功能测试报告怎么获取?

“双软认证”是指软件产品评估和软件企业评估&#xff0c;其中需要软件测试报告。 企业申请双软认证除了获得软件企业和软件产品的认证资质&#xff0c;同时也是对企业知识产权的一种保护方式&#xff0c;更可以让企业享受国家提供给软件行业的税收优惠政策。 那么&#xff0c;…

奇舞周刊第522期:“Vite 又开始搞事情了!!!”

奇舞推荐 ■ ■ ■ Vite 又开始搞事情了&#xff01;&#xff01;&#xff01; Vite 的最新版本将引入一种名为 Rolldown 的新型打包工具。 unocss 究竟比 tailwindcss 快多少&#xff1f; 我们知道 unocss 很快&#xff0c;也许是目前最快的原子化 CSS 引擎 (没有之一)。 巧用…

Flink:使用 Faker 和 DataGen 生成测试数据

博主历时三年精心创作的《大数据平台架构与原型实现&#xff1a;数据中台建设实战》一书现已由知名IT图书品牌电子工业出版社博文视点出版发行&#xff0c;点击《重磅推荐&#xff1a;建大数据平台太难了&#xff01;给我发个工程原型吧&#xff01;》了解图书详情&#xff0c;…

Linux 发布项目到OpenEuler虚拟机

后端&#xff1a;SpringBoot 前端&#xff1a;VUE3 操作系统&#xff1a;Linux 虚拟机&#xff1a;OpenEuler 发布项目是需要先关闭虚拟机上的防火墙 systemctl stop firewalld 一、运行后端项目到虚拟机 1、安装JDK软件包 查询Jdk是否已安装 dnf list installed | grep jd…

力扣每日一题 好子数组的最大分数 单调栈 双指针

Problem: 1793. 好子数组的最大分数 &#x1f496; 单调栈 思路 &#x1f468;‍&#x1f3eb; 参考题解 以当前高度为基准&#xff0c;寻找最大的宽度组成最大的矩形面积那就是要找左边第一个小于当前高度的下标left&#xff0c;再找右边第一个小于当前高度的下标right那宽…

Linux 磁盘的一生

注意&#xff1a;实验环境都是使用VMware模拟 ​ 磁盘接口类型这里vm中是SCSI&#xff0c;扩展sata,ide(有时间可以看看或者磁盘的历史) ​ 总结&#xff1a;磁盘从有到无—类似于建房子到可以住 ————————————————————————————————————…

【PHP + 代码审计】函数详解2.0

&#x1f36c; 博主介绍&#x1f468;‍&#x1f393; 博主介绍&#xff1a;大家好&#xff0c;我是 hacker-routing &#xff0c;很高兴认识大家~ ✨主攻领域&#xff1a;【渗透领域】【应急响应】 【Java、PHP】 【VulnHub靶场复现】【面试分析】 &#x1f389;点赞➕评论➕收…

【计算机网络篇】物理层(4)信道的极限容量,信道复用技术

文章目录 &#x1f354;信道的极限容量&#x1f6f8;造成信号失真的主要因素⭐码元的传输速率 &#x1f6f8;奈氏准则&#x1f6f8;香农公式&#x1f388;练习 &#x1f5d2;️小结 &#x1f354;信道复用技术⭐常见的信道复用技术&#x1f388;频分复用FDM&#x1f388;时分复…

Python之进程池、阻塞模式、非阻塞模式、进程间的通信、queue

非阻塞模式 # 当需要创建的子进程数量不多时&#xff0c;可以直接利用multiprocessing中的Process动态成生多个进程 # 但如果是上百甚至上千个目标&#xff0c;手动的去创建进程的工作量巨大&#xff0c;此时就可以用到multiprocessing模块提供的Pool方法. # 初始化Poo1时&…

分享5款专注于实用简洁的工具软件

​ 有时候一些小工具&#xff0c;能给你带来一些意想不到的效果&#xff0c;我们来看看下面这5款工具&#xff0c;你又用过其中几款呢&#xff1f; 1. 高效操作利器——Quicker ​ Quicker是一款旨在提高操作效率的强大工具。通过简单的自定义设置&#xff0c;用户能够创建个…

幼儿教育管理系统|基于jsp 技术+ Mysql+Java的幼儿教育管理系统设计与实现(可运行源码+数据库+设计文档)

推荐阅读100套最新项目 最新ssmjava项目文档视频演示可运行源码分享 最新jspjava项目文档视频演示可运行源码分享 最新Spring Boot项目文档视频演示可运行源码分享 2024年56套包含java&#xff0c;ssm&#xff0c;springboot的平台设计与实现项目系统开发资源&#xff08;可…

C++中的Union: 内存与类型转换技巧

在C中&#xff0c;union是一种特殊的数据类型&#xff0c;允许在相同的内存位置存储不同类型的数据。union提供了一种高效地利用内存的方式&#xff0c;但同时也要求开发者更加小心地处理数据以避免类型错误。 1. 基本定义 union定义了一个可以存储多种类型但任意时刻只能存储…

未来国家的希望在实体经济 民众的希望在投资理财

2024年经济的车轮已经滚滚而来&#xff0c;在阳春三月这个希望无限的季节&#xff0c;香港贵金属交易商、香港金银贸易场AA类147号行员金田金业认为&#xff0c;我们国家的希望在于实体经济发展&#xff0c;而民众要实现个人财务自由&#xff0c;仅仅靠打工还不够&#xff0c;更…