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

相关文章

MySQL的JDBC编程详解

《MySQL的JDBC编程详解》:本文主要介绍MySQL的JDBC编程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言一、前置知识1. 引入依赖2. 认识 url二、JDBC 操作流程1. JDBC 的写操作2. JDBC 的读操作总结前言本文介绍了mysq

java.sql.SQLTransientConnectionException连接超时异常原因及解决方案

《java.sql.SQLTransientConnectionException连接超时异常原因及解决方案》:本文主要介绍java.sql.SQLTransientConnectionExcep... 目录一、引言二、异常信息分析三、可能的原因3.1 连接池配置不合理3.2 数据库负载过高3.3 连接泄漏

Linux下MySQL数据库定时备份脚本与Crontab配置教学

《Linux下MySQL数据库定时备份脚本与Crontab配置教学》在生产环境中,数据库是核心资产之一,定期备份数据库可以有效防止意外数据丢失,本文将分享一份MySQL定时备份脚本,并讲解如何通过cr... 目录备份脚本详解脚本功能说明授权与可执行权限使用 Crontab 定时执行编辑 Crontab添加定

MySQL中On duplicate key update的实现示例

《MySQL中Onduplicatekeyupdate的实现示例》ONDUPLICATEKEYUPDATE是一种MySQL的语法,它在插入新数据时,如果遇到唯一键冲突,则会执行更新操作,而不是抛... 目录1/ ON DUPLICATE KEY UPDATE的简介2/ ON DUPLICATE KEY UP

MySQL分库分表的实践示例

《MySQL分库分表的实践示例》MySQL分库分表适用于数据量大或并发压力高的场景,核心技术包括水平/垂直分片和分库,需应对分布式事务、跨库查询等挑战,通过中间件和解决方案实现,最佳实践为合理策略、备... 目录一、分库分表的触发条件1.1 数据量阈值1.2 并发压力二、分库分表的核心技术模块2.1 水平分

Python与MySQL实现数据库实时同步的详细步骤

《Python与MySQL实现数据库实时同步的详细步骤》在日常开发中,数据同步是一项常见的需求,本篇文章将使用Python和MySQL来实现数据库实时同步,我们将围绕数据变更捕获、数据处理和数据写入这... 目录前言摘要概述:数据同步方案1. 基本思路2. mysql Binlog 简介实现步骤与代码示例1

使用shardingsphere实现mysql数据库分片方式

《使用shardingsphere实现mysql数据库分片方式》本文介绍如何使用ShardingSphere-JDBC在SpringBoot中实现MySQL水平分库,涵盖分片策略、路由算法及零侵入配置... 目录一、ShardingSphere 简介1.1 对比1.2 核心概念1.3 Sharding-Sp

MySQL 表空却 ibd 文件过大的问题及解决方法

《MySQL表空却ibd文件过大的问题及解决方法》本文给大家介绍MySQL表空却ibd文件过大的问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录一、问题背景:表空却 “吃满” 磁盘的怪事二、问题复现:一步步编程还原异常场景1. 准备测试源表与数据

Mac电脑如何通过 IntelliJ IDEA 远程连接 MySQL

《Mac电脑如何通过IntelliJIDEA远程连接MySQL》本文详解Mac通过IntelliJIDEA远程连接MySQL的步骤,本文通过图文并茂的形式给大家介绍的非常详细,感兴趣的朋友跟... 目录MAC电脑通过 IntelliJ IDEA 远程连接 mysql 的详细教程一、前缀条件确认二、打开 ID

MySQL的配置文件详解及实例代码

《MySQL的配置文件详解及实例代码》MySQL的配置文件是服务器运行的重要组成部分,用于设置服务器操作的各种参数,下面:本文主要介绍MySQL配置文件的相关资料,文中通过代码介绍的非常详细,需要... 目录前言一、配置文件结构1.[mysqld]2.[client]3.[mysql]4.[mysqldum