Mysql之AUTO_INCREMENT浅析

2023-11-03 05:38

本文主要是介绍Mysql之AUTO_INCREMENT浅析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • 术语
    • 锁模式
    • 注意点
    • 总结
    • 参考链接

AUTO_INCREMENT用于为表中的列设置一个自增序列,在非集群模式下,用它来为主键列自动生成值是一件很方便的事。并且,Mysql提供了一系列的锁机制来保证它的性能跟可靠性,通过这些锁机制,我们可以让它变得很高效。

术语

先来了解后面将要用到的术语。

  • Simple inserts

    能预先知道插入行数的语句。比如说单行插入(不包括INSERT … ON DUPLICATE KEY UPDATE),不带子句的多行插入(自增列不赋值或全赋值)。

  • Bulk inserts

    不能能预先知道插入行数的语句。比如INSERT … SELECT, REPLACE … SELECT 。这种模式下,InnoDB 会在处理时为每一行的自增列一次分配一个自增值

  • Mixed-mode inserts

    INSERT INTO t1 (c1,c2) VALUES (1,'a'), (NULL,'b'), (5,'c'), (NULL,'d');
    

    INSERT … ON DUPLICATE KEY UPDATE

  • Insert-like

    以上所有插入语句的统称

锁模式

锁模式在启动时通过innodb_autoinc_lock_mode这个变量来配置,它有3个值:0, 1, 2。分别对应traditional(传统)、consecutive(连续)、interleaved(交错)3种模式。

Mysql5.6~5.7里,这个配置项的默认值是1,从Mysql8开始,它的默认值2。这个一方面是因为模式2的性能更好,另一方面是因为从Mysql8开始,默认的主从复制的方式由statement-based 改为了row basedrow based 能保证innodb_autoinc_lock_mode=2时主从复制时的数据不会出现不一致的问题。好了,下面开始详细了解这三种锁模式吧!

  • traditional

    在这种模式下,Mysql所有的Insert-like操作都会设置一个表级别的AUTO-INC 锁,这个锁会在单条语句执行完毕时释放。

    也就是说,如果有多个事务同时对同一张表执行Insert-like 操作,那么,即使它们没有操作同一条记录,也会串行执行。所以,它的性能相对另外两种模式来说会比较糟糕。

  • consecutive

    在这种模式下,对于Simple inserts语句,Mysql会在语句执行的初始阶段将一条语句需要的所有自增值会一次性分配出来,并且通过设置一个互斥量来保证自增序列的一致性,一旦自增值生成完毕,这个互斥量会立即释放,不需要等到语句执行结束。

    所以,在consecutive模式,多事务并发执行Simple inserts这类语句时, 相对traditional模式,性能会有比较大的提升。

    由于一开始就为语句分配了所有需要的自增值,那么对于像Mixed-mode inserts这类语句,就有可能多分配了一些值给它,从而导致自增序列出现"空隙"。而traditional模式因为每一次只会为一条记录分配自增值,所以不会有这种问题。

    另外,对于Bulk inserts语句,依然会采取AUTO-INC锁。所以,如果有一条Bulk inserts语句正在执行的话,Simple inserts也必须等到该语句执行完毕才能继续执行。

  • interleaved

    在这种模式下,对于所有的Insert-like语句,都不会存在表级别的AUTO-INC锁,意味着同一张表上的多个语句并发时阻塞会大幅减少。

    但是,这种模式必须运行在row based或者mixed-format(其实说白了也是row based,只不过mysql会根据自增模式为不安全的语句自动选择row based模式)复制模式下。因为interleaved这种模式下对于相同顺序的语句每次执行行都会产生不同的结果(谁竞争到互斥量,谁就能获得生成自增值的权利),所以,如果复制模式是statement-based或者通过binlog进行数据恢复时(牵扯到binlog的语句重放),可能会导致数据不一致。

注意点

  • 自增值的生成后是不能回滚的,所以自增值生成后,事务回滚了,那么那些已经生成的自增值就丢失了,从而使自增列的数据出现空隙

  • 正常情况下,自增列是不存在0这个值的。所以,如果插入语句中对自增列设置的值为0或者null,就会自动应用自增序列。

    那么,如果想在自增列中插入为0这个值,怎么办呢?可以通过将SQL Mode设置为NO_AUTO_VALUE_ON_ZERO即可

  • 在Mysql5.7以及更早之前,自增序列的计数器(auto-increment counter)是保存在内存中的。auto-increment counter在每次Mysql重新启动后通过类似下面的这种语句进行初始化:

    SELECT MAX(AUTO_INC_COLUMN) FROM table_name FOR UPDATE
    

    而从Mysql8开始,auto-increment counter被存储在了redo log中,并且每次变化都会刷新到redo log中。另外,我们可以通过ALTER TABLE … AUTO_INCREMENT = N 来主动修改
    auto-increment counter

总结

  1. 单实例下,可以设置innodb_autoinc_lock_mode=2
  2. 主从
    1. 复制模式为statement-based,设置innodb_autoinc_lock_mode=1
    2. 复制模式为row based或者mixed-format,设置innodb_autoinc_lock_mode=2

参考链接

  • 15.6.1.6 AUTO_INCREMENT Handling in InnoDB
  • 15.18.2 InnoDB Recovery
  • 17.5.1.1 Replication and AUTO_INCREMENT
  • 3.6.9 Using AUTO_INCREMENT

这篇关于Mysql之AUTO_INCREMENT浅析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/335958

相关文章

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

使用 sql-research-assistant进行 SQL 数据库研究的实战指南(代码实现演示)

《使用sql-research-assistant进行SQL数据库研究的实战指南(代码实现演示)》本文介绍了sql-research-assistant工具,该工具基于LangChain框架,集... 目录技术背景介绍核心原理解析代码实现演示安装和配置项目集成LangSmith 配置(可选)启动服务应用场景

oracle DBMS_SQL.PARSE的使用方法和示例

《oracleDBMS_SQL.PARSE的使用方法和示例》DBMS_SQL是Oracle数据库中的一个强大包,用于动态构建和执行SQL语句,DBMS_SQL.PARSE过程解析SQL语句或PL/S... 目录语法示例注意事项DBMS_SQL 是 oracle 数据库中的一个强大包,它允许动态地构建和执行

SQL 中多表查询的常见连接方式详解

《SQL中多表查询的常见连接方式详解》本文介绍SQL中多表查询的常见连接方式,包括内连接(INNERJOIN)、左连接(LEFTJOIN)、右连接(RIGHTJOIN)、全外连接(FULLOUTER... 目录一、连接类型图表(ASCII 形式)二、前置代码(创建示例表)三、连接方式代码示例1. 内连接(I

在MySQL执行UPDATE语句时遇到的错误1175的解决方案

《在MySQL执行UPDATE语句时遇到的错误1175的解决方案》MySQL安全更新模式(SafeUpdateMode)限制了UPDATE和DELETE操作,要求使用WHERE子句时必须基于主键或索引... mysql 中遇到的 Error Code: 1175 是由于启用了 安全更新模式(Safe Upd

浅析如何使用Swagger生成带权限控制的API文档

《浅析如何使用Swagger生成带权限控制的API文档》当涉及到权限控制时,如何生成既安全又详细的API文档就成了一个关键问题,所以这篇文章小编就来和大家好好聊聊如何用Swagger来生成带有... 目录准备工作配置 Swagger权限控制给 API 加上权限注解查看文档注意事项在咱们的开发工作里,API

轻松上手MYSQL之JSON函数实现高效数据查询与操作

《轻松上手MYSQL之JSON函数实现高效数据查询与操作》:本文主要介绍轻松上手MYSQL之JSON函数实现高效数据查询与操作的相关资料,MySQL提供了多个JSON函数,用于处理和查询JSON数... 目录一、jsON_EXTRACT 提取指定数据二、JSON_UNQUOTE 取消双引号三、JSON_KE

MySql死锁怎么排查的方法实现

《MySql死锁怎么排查的方法实现》本文主要介绍了MySql死锁怎么排查的方法实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录前言一、死锁排查方法1. 查看死锁日志方法 1:启用死锁日志输出方法 2:检查 mysql 错误

MySQL数据库函数之JSON_EXTRACT示例代码

《MySQL数据库函数之JSON_EXTRACT示例代码》:本文主要介绍MySQL数据库函数之JSON_EXTRACT的相关资料,JSON_EXTRACT()函数用于从JSON文档中提取值,支持对... 目录前言基本语法路径表达式示例示例 1: 提取简单值示例 2: 提取嵌套值示例 3: 提取数组中的值注意

MySQL修改密码的四种实现方式

《MySQL修改密码的四种实现方式》文章主要介绍了如何使用命令行工具修改MySQL密码,包括使用`setpassword`命令和`mysqladmin`命令,此外,还详细描述了忘记密码时的处理方法,包... 目录mysql修改密码四种方式一、set password命令二、使用mysqladmin三、修改u