原文地址:http://www.jbxue.com/db/12819.html
在mysql数据库中,使用auto_increment 字段来辅助为自增列赋值。
显示有关msyql自增ID的相关配置信息:
输出:
auto_increment_increment 1
auto_increment_offset 1
说明:
auto_increment_increment = 1 ,每次插入数据,id++;auto_increment_offset =1 ,说明从1开始。
一般情况下auto_increment_increment=1,auto_increment_offset=1.是默认初始值。
可以在my.cnf文件中进行重新指定。auto_increment满足一个数学当纳法推导公式:
value=auto_increment_increment*N+auto_increment_offset
表中有auto_increment列,innodb用一种锁策略来保正这个auto_increment列值。
如果创建一个表,指定了auto_increment表。那么innodb会维护一个auto_increment计数器。
当访问这个计数器,innodb会使用表级锁来锁定这访问过程,这个发生在分析auto_increment
值的过程中,而不是在一个执行事务中。也就是说:这个访问发生在事务执行前面。
所以和这个锁不是发生整个事务过程中,仅仅是分析语句判断出auto_increment这一过程中。
auto_increment计数器,一直维护在内存中,当server重启或是停止后重启,innodb会为每个表初始化这个计数器。
对一个含有自增列(通常为id)的表执行
mysql> SHOW CREATE TABLE table_name;
输出:
其中该状态下,表中AUTO_INCREMENT 字段值为704677 ,说明下一个插入的id 为704677 ,同时插入成功后,AUTO_INCREMENT=704678。
mysql 数据库中的id设为自增,容易产生id不连续的问题。
有时将一个表中的所有数据清除,但是在插入数据时,id还是在未清除前的基础上累加。
需要解决如下问题:
1. 删除数据,同时要求再次插入数据时,id从1 开始累计
分析:
Truncate table 表名 速度快,而且效率高,因为:TRUNCATE TABLE 在功能上与不带 WHERE 子句的 DELETE 语句相同:二者均删除表中的全部行。
但是,TRUNCATE TABLE 比 DELETE 速度快,且使用的系统和事务日志资源少。DELETE 语句每次删除一行,并在事务日志中为所删除的每行记录一项。TRUNCATE TABLE 通过释放存储表数据所用的数据页来删除数据,并且只在事务日志中记录页的释放。
TRUNCATE TABLE 删除表中的所有行,但表结构及其列、约束、索引等保持不变。新行标识所用的计数值重置为该列的种子。如果想保留标识计数值,请改用 DELETE。如果要删除表定义及其数据,请使用 DROP TABLE 语句。
2,查看当前状态下一个表的auto_increment 值:
SHOW TABLE STATUS LIKE 'table_name'
输出:
可以查看该表当前的Auto_increment的值。
3,对有些失败插入操作,如果不是语法错误,比如重复的唯一键值,也会造成自增id的不连续。
4,mysql5.1.22之前,innodb使用一个表锁解决自增字段的一致性问题(内部是用一个计数器维护,每次自增时要加表锁),如果一行一行的插入数据则没有什么问题,但是如果大量的并发插入就废了,表锁会引起SQL堵塞,不但影响效率,而且可能会瞬间达到max_connections而崩溃。
在 5.1.22之后,innodb使用新的方式解决自增字段一致性问题,对于可以预判行数的insert语句,innodb使用一个轻量级的互斥量。如:某一insert语句1执行前,表的AUTO_INCREMENT=1,语句1的插入行数已知为3,innodb在语句1的实际插入操作执行前就预分配给该语句三个自增值,当有一个新的insert语句2要执行时,读取的AUTO_INCREMENT=4,这样虽然语句1可能还没有执行完,语句2就可直接执行无需等待语句2。
这种方式对于可预判插入行数的插入语句有效,如:insert和replace。对于无法提前获知插入行数的语句,如:insert...select...、replace...select...和load data则innodb还是使用表锁。
insert语句中有时会显示的设置自增字段的值,对于这种情况innodb还是会预分配给语句总行数的自增值而不是只有实际使用系统自增的行。因而有可能会造成自增字段的值不连续。
例如:
INSERT INTO t1 (c1,c2) VALUES (1,'a'), (NULL,'b'), (5,'c'), (NULL,'d');
c1 为自增字段,实际的AUTO_INCREMENT 会大2。
设置新自增互斥方式:
通过配置选项:innodb_autoinc_lock_mode,它是专门用来在使用auto_increment的情况下调整锁策略的。
目前有三种选择:
在mysql数据库中,使用auto_increment 字段来辅助为自增列赋值。
显示有关msyql自增ID的相关配置信息:
输出:
auto_increment_increment 1
auto_increment_offset 1
说明:
auto_increment_increment = 1 ,每次插入数据,id++;auto_increment_offset =1 ,说明从1开始。
一般情况下auto_increment_increment=1,auto_increment_offset=1.是默认初始值。
可以在my.cnf文件中进行重新指定。auto_increment满足一个数学当纳法推导公式:
value=auto_increment_increment*N+auto_increment_offset
表中有auto_increment列,innodb用一种锁策略来保正这个auto_increment列值。
如果创建一个表,指定了auto_increment表。那么innodb会维护一个auto_increment计数器。
当访问这个计数器,innodb会使用表级锁来锁定这访问过程,这个发生在分析auto_increment
值的过程中,而不是在一个执行事务中。也就是说:这个访问发生在事务执行前面。
所以和这个锁不是发生整个事务过程中,仅仅是分析语句判断出auto_increment这一过程中。
auto_increment计数器,一直维护在内存中,当server重启或是停止后重启,innodb会为每个表初始化这个计数器。
对一个含有自增列(通常为id)的表执行
mysql> SHOW CREATE TABLE table_name;
输出:
其中该状态下,表中AUTO_INCREMENT 字段值为704677 ,说明下一个插入的id 为704677 ,同时插入成功后,AUTO_INCREMENT=704678。
mysql 数据库中的id设为自增,容易产生id不连续的问题。
有时将一个表中的所有数据清除,但是在插入数据时,id还是在未清除前的基础上累加。
需要解决如下问题:
1. 删除数据,同时要求再次插入数据时,id从1 开始累计
分析:
Truncate table 表名 速度快,而且效率高,因为:TRUNCATE TABLE 在功能上与不带 WHERE 子句的 DELETE 语句相同:二者均删除表中的全部行。
但是,TRUNCATE TABLE 比 DELETE 速度快,且使用的系统和事务日志资源少。DELETE 语句每次删除一行,并在事务日志中为所删除的每行记录一项。TRUNCATE TABLE 通过释放存储表数据所用的数据页来删除数据,并且只在事务日志中记录页的释放。
TRUNCATE TABLE 删除表中的所有行,但表结构及其列、约束、索引等保持不变。新行标识所用的计数值重置为该列的种子。如果想保留标识计数值,请改用 DELETE。如果要删除表定义及其数据,请使用 DROP TABLE 语句。
2,查看当前状态下一个表的auto_increment 值:
SHOW TABLE STATUS LIKE 'table_name'
输出:
可以查看该表当前的Auto_increment的值。
3,对有些失败插入操作,如果不是语法错误,比如重复的唯一键值,也会造成自增id的不连续。
4,mysql5.1.22之前,innodb使用一个表锁解决自增字段的一致性问题(内部是用一个计数器维护,每次自增时要加表锁),如果一行一行的插入数据则没有什么问题,但是如果大量的并发插入就废了,表锁会引起SQL堵塞,不但影响效率,而且可能会瞬间达到max_connections而崩溃。
在 5.1.22之后,innodb使用新的方式解决自增字段一致性问题,对于可以预判行数的insert语句,innodb使用一个轻量级的互斥量。如:某一insert语句1执行前,表的AUTO_INCREMENT=1,语句1的插入行数已知为3,innodb在语句1的实际插入操作执行前就预分配给该语句三个自增值,当有一个新的insert语句2要执行时,读取的AUTO_INCREMENT=4,这样虽然语句1可能还没有执行完,语句2就可直接执行无需等待语句2。
这种方式对于可预判插入行数的插入语句有效,如:insert和replace。对于无法提前获知插入行数的语句,如:insert...select...、replace...select...和load data则innodb还是使用表锁。
insert语句中有时会显示的设置自增字段的值,对于这种情况innodb还是会预分配给语句总行数的自增值而不是只有实际使用系统自增的行。因而有可能会造成自增字段的值不连续。
例如:
INSERT INTO t1 (c1,c2) VALUES (1,'a'), (NULL,'b'), (5,'c'), (NULL,'d');
c1 为自增字段,实际的AUTO_INCREMENT 会大2。
设置新自增互斥方式:
通过配置选项:innodb_autoinc_lock_mode,它是专门用来在使用auto_increment的情况下调整锁策略的。
目前有三种选择: