MySQL innodb_autoinc_lock_mode 详解

news/2024/7/5 4:02:38

innodb_autoinc_lock_mode这个参数控制着在向有auto_increment 列的表插入数据时,相关锁的行为;

通过对它的设置可以达到性能与安全(主从的数据一致性)的平衡

【0】我们先对insert做一下分类

  首先insert大致上可以分成三类:
    1、simple insert 如insert into t(name) values('test')
    2、bulk insert 如load data | insert into ... select .... from ....
    3、mixed insert 如insert into t(id,name) values(1,'a'),(null,'b'),(5,'c');

 

【1】innodb_autoinc_lock_mode 的说明

  innodb_auto_lockmode有三个取值:
    1、0 这个表示tradition 传统
    2、1 这个表示consecutive 连续
    3、2 这个表示interleaved 交错

【1.1】tradition(innodb_autoinc_lock_mode=0) 模式:

  1、它提供了一个向后兼容的能力
  2、在这一模式下,所有的insert语句("insert like") 都要在语句开始的时候得到一个
     表级的auto_inc锁,在语句结束的时候才释放这把锁,注意呀,这里说的是语句级而不是事务级的,
       一个事务可能包涵有一个或多个语句。
  3、它能保证值分配的可预见性,与连续性,可重复性,这个也就保证了insert语句在复制到slave
          的时候还能生成和master那边一样的值(它保证了基于语句复制的安全)。
     4、由于在这种模式下auto_inc锁一直要保持到语句的结束,所以这个就影响到了并发的插入。

 

【1.2】consecutive(innodb_autoinc_lock_mode=1) 模式:

  1、这一模式下去simple insert 做了优化,由于simple insert一次性插入值的个数可以立马得到
          确定,所以mysql可以一次生成几个连续的值,用于这个insert语句;总的来说这个对复制也是安全的
          (它保证了基于语句复制的安全)
  2、这一模式也是mysql的默认模式,这个模式的好处是auto_inc锁不要一直保持到语句的结束,只要
          语句得到了相应的值后就可以提前释放锁

 

【1.3】interleaved(innodb_autoinc_lock_mode=2) 模式
  1、由于这个模式下已经没有了auto_inc锁,所以这个模式下的性能是最好的;但是它也有一个问题,就是
          对于同一个语句来说它所得到的auto_incremant值可能不是连续的。

 

【2】如果你的二进制文件格式是mixed | row 那么这三个值中的任何一个对于你来说都是复制安全的。

  由于现在mysql已经推荐把二进制的格式设置成row,所以在binlog_format不是statement的情况下最

  好是innodb_autoinc_lock_mode=2 这样可能知道更好的性能。

 

最后以一个关于auto_increment 的例子来结束

例子:不要没事去更新一个auto_increment 列的值

第一步:重现一下场景

create table t(x int auto_increment not null primary key);
insert into t(x) values(0),(null),(3);
select * from t;
+---+
| x |
+---+
| 1 |
| 2 |
| 3 |
+---+

第二步:重现一下引发问题的SQL

update t set x=4 where x=1;
select * from t;
+---+
| x |
+---+
| 2 |
| 3 |
| 4 |
+---+

第三步:重现一下总是的表现形式

insert into t(x) values(0);
ERROR 1062 (23000): Duplicate entry '4' for key 'PRIMARY'

第四步:对问题的总结

  执行完第一步的时候mysql知道下一个auto_increment值是4。

  执行完第二步的时候mysql并不知道4已经被人为的占用了,所以执行第三步的时候就出错了。

 

---

交流学习&打赏

 


http://lihuaxi.xjx100.cn/news/241571.html

相关文章

全国首个窄带物联网实验局落户福州 助力智慧城市建设

市政府、省经信委、华为技术有限公司、智润科技有限公司在榕签署NB-IoT(窄带物联网)项目合作备忘录,合力在福州推进完成全国首个NB-IoT实验局建设和商业化部署,建立开放实验室,共同打造福建NB-IoT物联网应用样板点&…

CCF201503-4 网络延时(100分)

试题编号: 201503-4 试题名称: 网络延时 时间限制: 1.0s 内存限制: 256.0MB 问题描述: 问题描述给定一个公司的网络,由n台交换机和m台终端电脑组成,交换机与交换机、交换机与电脑之间使用网络连…

python 列表维度_如何输出python中list的维度

python中输出list的维度可以使用numpy来实现:import numpy as np a = [[1,2],[3,4]] print(np.array(a).shape) 扩展: reshape&resize&shape改变数组维度 reshape函数:不改变原数组维度,有返回值 resize函数:直接改变原数组维度,无返回值 shape属性:直接改变原数…

固态硬盘驱动器在设计上有个安全漏洞 易导致数据损毁

近年来,出于对提升系统运行速度的渴求,传统机械硬盘(HDD)的市场正被更高速的固态硬盘驱动器(SSD)所蚕食。尽管很多用户仍在采用 SSD 系统盘 HDD 仓库盘的组合,但后者被淘汰也只是时间问题。然而…

Parcelable与Serializable的比较

Parcel: Android中的序列化方式,可用于跨进程传输 Parcelable 进程间 如:想从一个第三方app拿进程回来 Serializable 进程内

我眼中的DevOps(转)

过去一年以来,一批来自欧美的、不墨守陈规的系统管理员和开发人员一直在谈论一个新概念:DevOps。DevOps 就是开发(Development)和运维(Operations)这两个领域的合并。(如果没错的话,…

nodejs 截断字符串_JS截取字符串常用方法详细整理

使用 substring()或者slice()函数:split()功能:使用一个指定的分隔符把一个字符串分割存储到数组例子:str”jpg|bmp|gif|ico|png”;arrtheString.split(”|”);//arr是一个包含字符值”jpg”、”bmp”、”gif”、”ico”和”png”的数组函数&…