当前位置:网站首页>What if the self incrementing ID of online MySQL is exhausted?
What if the self incrementing ID of online MySQL is exhausted?
2022-07-04 18:36:00 【Program ape DD_】
author : Junk programmer link :https://www.jianshu.com/p/a6bc14005b52
MySQL Self increasing of id They all define initial values , And then step by step . Although there is no upper limit for natural numbers , But it defines the length of bytes that represent this number , Computer storage has an upper limit . such as , Unsigned integer (unsigned int) yes 4 Bytes , The upper limit is 2^32 - 1. That's growing id run out , What will happen? ?
data:image/s3,"s3://crabby-images/7e657/7e65796b0931f6bbb5775a2664312faa90f11ec5" alt="d12a3bdca8bfb85ecb7b7af52f2c5de3.jpeg"
Table definition self increment id
The logic of the table definition after the increment reaches the upper limit is : Apply for the next one id when , The resulting value remains unchanged .
mysql> create table t(id int unsigned auto_increment primary key) auto_increment=4294967295;
Query OK, 0 rows affected (0.01 sec)
mysql> insert into t values(null);
Query OK, 1 row affected (0.00 sec)
mysql> show create table t;
+-------+-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
| Table | Create Table |
+-------+-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
| t | CREATE TABLE `t` (
`id` int unsigned NOT NULL AUTO_INCREMENT,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=4294967295 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci |
+-------+-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
1 row in set (0.00 sec)
// Successfully inserted a line 4294967295
mysql> insert into t values(null);
ERROR 1062 (23000): Duplicate entry '4294967295' for key 't.PRIMARY'
first insert After success , Of this table AUTO_INCREMENT still 4294967295, Leading to the second insert And get the same self increase id value , Try to execute the insert statement again , Primary key conflict .
2^32 - 1(4294967295) Not a particularly large number , A table with frequent inserts and deletions is likely to run out . When creating a table, you need to consider whether it is possible for your table to reach the upper limit , If you have any , It should be created as 8 Bytes of bigint unsigned.
InnoDB The system is self increasing row_id
If you create InnoDB Table does not specify primary key , be InnoDB Will automatically create an invisible ,6 Bytes of row_id.InnoDB Maintains a global dict_sys->row_id value
data:image/s3,"s3://crabby-images/ee05d/ee05df4e752acf144cdb72400c2d463612483498" alt="867e86dff5dfa929d26495a66f74b098.jpeg"
All without primary key InnoDB surface , Insert each row of data , Will be the current dict_sys->row_id As the... To insert data row_id, And then put dict_sys->row_id Add 1.
When the code is implemented row_id The length is 8 An unsigned long integer of bytes (bigint unsigned). but InnoDB In design , to row_id All that's left is 6 The length of bytes , In this way, only the last 6 Bytes , therefore row_id Values that can be written to the data table , There are two characteristics :
row_id Write the range of values in the table , It's from 0 To 2^48 - 1
When dict_sys.row_id=2^48 when , If you want to apply for inserting data again row_id, Take the last after you get it 6 A byte is 0
Write to the table row_id from 0~2^48 - 1. When the limit is reached , The next value is 0, And then the cycle goes on .
2^48 - 1 It's already big , But if one MySQL Examples live a long time , It's still possible to reach the upper limit .
InnoDB in , succeed in applying for sth. row_id=N after , Write this row of data to the table ; If the table already exists row_id=N The line of , The newly written line will cover the original line .
Verify the conclusion : adopt gdb Modify the auto increment of the system row_id. use gdb It's for the sake of reproducing the problem , Can only be used in a test environment .
row_id Used up validation sequence
data:image/s3,"s3://crabby-images/eac21/eac21ad5088188db1acb0410d053d20a962e2f85" alt="6247521f90e5eadd6ea60c4120639b3b.jpeg"
row_id Verification of the effect of using up
data:image/s3,"s3://crabby-images/319bc/319bc58a043f4a489bd76c2c1739a7f3ef5a4562" alt="01066a38c8ef8334eba76185372d1d04.jpeg"
so , For me gdb take dict_sys.row_id Set to 2^48 after , Insert again a=2 It will appear in the table t The first line of , Because the value of row_id=0.
Then insert a=3, because row_id=1, It covers the previous a=1 The line of , because a=1 In this line row_id It's also 1.
So it should be InnoDB Create auto incrementing primary key in the table : When the table increases id After reaching the limit , The primary key conflict error will be reported when inserting data again .
After all, overlay data , It means data loss , Affect data reliability ; It's a conflict , Insert the failure , Impact availability . General reliability is better than availability .
Xid
redo log and binlog There's a common field Xid, Used to correspond to transactions .Xid stay MySQL How is the interior generated ?
MySQL A global variable is maintained internally global_query_id
data:image/s3,"s3://crabby-images/fffef/fffef3379f2e4181eafac64a1bf2dfe2d1e99e4b" alt="eb2b4b1b215ad800938f4ed99a99828d.jpeg"
Every time a statement is executed , Assign it to query_id, Then give the variable +1:
data:image/s3,"s3://crabby-images/07454/0745441ff228a1171a9f709769ed28465db8fafc" alt="172d0f0ddd9174db402517d6bd752450.jpeg"
If the current statement is the first statement executed by the transaction , be MySQL And at the same time query_id Assigned to the transaction Xid:
data:image/s3,"s3://crabby-images/705ef/705ef763932efd736e97df354240dca087a5ba16" alt="162d693b80bc6377b762af73f88ca308.jpeg"
and global_query_id Is a pure memory variable , It's cleared after restart . So the same DB example , Different business Xid Could be the same .
but MySQL After restart, a new... Will be regenerated binlog file , This guarantees the same binlog In the file Xid only .
although MySQL Rebooting doesn't result in the same binlog There are two identical Xid, But if global_query_id Reach the upper limit , Will continue from 0 Start counting . In theory, there will still be the same binlog It's the same Xid.
because global_query_id8 byte , ceiling 2^64 - 1. If this happens , Need to meet :
Execute a transaction , hypothesis Xid yes A
Next 2^64 Second query statement , Give Way global_query_id go back to A 2^64 It's too big , This possibility exists only in theory .
Start another transaction , This business is Xid It's also A
Innodb trx_id
Xid from server Layer maintenance
InnoDB For internal use Xid, For connection InnoDB Business and server
but InnoDB Their own trx_id, It's another maintenance business id(transaction id).
InnoDB One was maintained internally max_trx_id Global variables , Each time you need to apply for a new trx_id when , You get max_trx_id The current value of the , And then max_trx_id Add 1.
InnoDB The core idea of data visibility
Each row of data records the update to it trx_id, When a transaction reads a row of data , Determine if the data is visible , It's through the consistency view of the transaction and this row of data trx_id comparing .
For executing transactions , You can start your information_schema.innodb_trx You can see in the table that trx_id.
Look at the following case : The transaction trx_id
S2 The execution record of :
mysql> use information_schema;
Reading table information for completion of table and column names
You can turn off this feature to get a quicker startup with -A
Database changed
mysql> select trx_id, trx_mysql_thread_id from innodb_trx;
+-----------------+---------------------+
| trx_id | trx_mysql_thread_id |
+-----------------+---------------------+
| 421972504382792 | 70 |
+-----------------+---------------------+
1 row in set (0.00 sec)
mysql> select trx_id, trx_mysql_thread_id from innodb_trx;
+---------+---------------------+
| trx_id | trx_mysql_thread_id |
+---------+---------------------+
| 1355623 | 70 |
+---------+---------------------+
1 row in set (0.01 sec)
S2 from innodb_trx The two fields found in the table , Second field trx_mysql_thread_id The thread id. Show threads id, This is to show the thread corresponding to the transaction seen in the two queries id All are 5, namely S1 Thread .
t2 It shows trx_id It's a big number ;t4 All the time trx_id yes 1289, It looks like a normal number . Why is this ?
t1 when ,S1 No updates yet , It's a read-only transaction . For read-only transactions ,InnoDB It doesn't allocate trx_id:
t1 when ,trx_id The value is 0. And this big number , Just for display
until S1 stay t3 When the insert,InnoDB To really distribute trx_id. therefore t4 when ,S2 We found that trx_id The value is 1289.
In addition to the obvious modification of class statements , If in select Add... After the statement for update, It's not a read-only transaction .
update and delete Statement in addition to the transaction itself , It also involves marking old data for deletion , That is to put the data in purge Waiting in the queue for subsequent physical deletion , This operation will also put max_trx_id+1, So add at least 2
InnoDB The background operation of , For example, such operations as table index information statistics , It will also start internal affairs , So you might see ,trx_id The value is not in accordance with the plus 1 Incremental .
t2 How did the big numbers come from ?
Every time I look up , It is calculated temporarily by the system :
Of the current transaction trx The pointer address of a variable is converted to an integer , Plus 248
This ensures :
Because the same read-only transaction is during execution , Its pointer address doesn't change , So no matter in innodb_trx Still innodb_locks table , The same read-only transaction trx_id It's going to be the same
If there are parallel read-only transactions , Of every business trx The pointer address of the variable must be different . such , Different concurrent read-only transactions , Found out trx_id It's different .
Why add 248?
Ensure that read-only transactions are displayed trx_id It's worth more , Under normal circumstances, it will be different from read-write transactions id. but trx_id Follow row_id The logic of is similar to , Defined as 8 Bytes . Theoretically, it is possible to display a read-write transaction and a read-only transaction trx_id identical . But the probability is very low , There's no real harm , No matter .
Why read only transactions are not allocated trx_id?
Reduce the size of the active transaction array in the transaction view . Because the currently running read-only transaction , It doesn't affect the visibility judgment of data . therefore , When creating a consistent view of a transaction ,InnoDB You just need to copy the read-write transaction trx_id
Reduce trx_id The number of applications .InnoDB Perform a normal select sentence , Also corresponding to a read-only transaction . So after optimizing read-only transactions , Ordinary query statements do not need to apply for trx_id, Greatly reduce concurrent transaction requests trx_id Lock conflict
Because read only transactions are not allocated trx_id, obviously trx_id It's slowing down .
but max_trx_id Persistent storage , Rebooting doesn't reset to 0. Theoretically , As long as a MySQL I ran long enough , It's possible max_trx_id achieve 2^48 - 1, And then from 0 Start the cycle .
After reaching this state ,MySQL A dirty read will continue to appear bug:
First of all, put the current max_trx_id Change it to 2^48 - 1. This is repeatable .
Repeat dirty reading
data:image/s3,"s3://crabby-images/03797/037973ea0da4cc74c274935de41283130bb51022" alt="f1ac332b3754e826b480a8c1544ef193.jpeg"
data:image/s3,"s3://crabby-images/23035/23035a535dc69f214b359957411b4eaf8e4601ee" alt="9c038d88f14be1aec5c562697b393279.jpeg"
Because of the systematic max_trx_id Is set to 2^48 - 1, So in session A Transactions started TA The lowest water level in the world is 2^48 - 1.
t2 when :
session B Carry out article 1 update Statement transactions id=2^48 - 1
Second thing id Namely 0 了 , This article update On the data version generated after execution trx_id=0
t3 when :
session A perform select Visibility judgment of :c=3 This data version of trx_id(0), Less than transaction TA The lowest water level in the world (2^48 - 1), So we think that the data is visible .
But it's dirty reading .
Because the low water level will continue to increase , And the business id from 0 Start counting , Cause the system after that moment , All queries are dirty read .
also MySQL Restart time max_trx_id I don't know 0, The restart MySQL, This bug There is still . What about this bug It only exists in theory ?
Suppose a MySQL Example of TPS yes 50w, Keep doing this ,17.8 That's going to happen in a few years . But from MySQL It's really popular until now , I'm afraid none of them have reached the upper limit yet . however , as long as MySQL Instance service time is long enough , There is bound to be bug.
It can also deepen the understanding of low water levels and data visibility
thread_id
The system holds a global variable thread_id_counter
data:image/s3,"s3://crabby-images/f0a22/f0a2221915d0ed8706710ae42e18353cad2bd331" alt="0ca5d51e42d657ade77a46053276c5d9.jpeg"
Every new connection , will thread_id_counter The thread variable assigned to this new connection new_id.
thread_id_counter Defined as 4 Bytes , Therefore, the 2^32 - 1, It will reset to 0, Continue to increase .
data:image/s3,"s3://crabby-images/18b5a/18b5a8d80a05abace165414eb79098a8604a9a14" alt="eb07f2ad59367b448b6d103cff6823f8.jpeg"
But not in show processlist See two identical thread_id. because MySQL Used a unique array
data:image/s3,"s3://crabby-images/4c5e2/4c5e274501108296003c41caf783cff7da26ece9" alt="bcdcaa535a2084d05e919fbb77a13b9b.jpeg"
Assign... To a new thread thread_id The logic of time :
data:image/s3,"s3://crabby-images/02d74/02d741cea419265fb66ca84bd3ea69fdbec4ff9e" alt="5e467776fb4d557857caf8fbc7346ba3.jpeg"
summary
Every kind of self increasing id There are different application scenarios , The performance after reaching the upper limit is also different :
On the table id When the limit is reached , When you apply again, its value will not change , This leads to a primary key conflict error when you continue to insert data
row_id When the limit is reached , You will return 0 Increase again , If the same row_id, The later data will cover the previous data
Xid Just not in the same binlog Duplicate values appear in the file . Although in theory there will be repeated values , But the probability is very small , Negligible
InnoDB Of max_trx_id Every time MySQL Restart will be saved , So the example of dirty reading mentioned in our article is a necessary one bug, Fortunately, we still have plenty of time
We have created a high-quality technical exchange group , With good people , I will be excellent myself , hurriedly Click Add group , Enjoy growing up together . in addition , If you want to change jobs recently , Years ago, I spent 2 A wave of large factory classics were collected in a week , Those who are ready to change jobs after the festival can Click here to get !
Recommended reading
For resistance 7-Zip, List “ Three sins ” ? Net friend :“ The first 3 That's the point ?”
Please , Stop brushing Star 了 ! This one “ patriotic ” No problem !
··································
Hello , I'm a procedural ape DD,10 Old driver developed in 、 Alibaba cloud MVP、 Tencent cloud TVP、 I have published books and started a business 、 State-owned enterprises 4 In the Internet 6 year . From ordinary developers to architects 、 Then to the partner . Come all the way , My deepest feeling is that I must keep learning and pay attention to the frontier . As long as you can hold on , Think more 、 Don't complain 、 Do it frequently , It's easy to overtake on a curve ! therefore , Don't ask me what I'm doing now, whether it's in time . If you are optimistic about one thing , It must be persistence to see hope , Instead of sticking to it when you see hope . believe me , Just stick to it , You must be better than now ! If you don't have any direction , You can pay attention to me first , Some cutting-edge information is often shared here , Help you accumulate the capital to overtake on the curve .
边栏推荐
- 【Go语言刷题篇】Go完结篇|函数、结构体、接口、错误入门学习
- Tutorial on the use of Huawei cloud modelarts (with detailed illustrations)
- Performance test of Gatling
- LD_LIBRARY_PATH 环境变量设置
- Unity makes revolving door, sliding door, cabinet door drawer, click the effect of automatic door opening and closing, and automatically play the sound effect (with editor extension code)
- People in the workplace with a miserable expression
- 【210】PHP 定界符的用法
- DB engines database ranking in July 2022: Microsoft SQL Server rose sharply, Oracle fell sharply
- Achieve animation effect through event binding
- 俄罗斯 Arenadata 发布基于PostgreSQL的产品
猜你喜欢
五千字讲清楚团队自组织建设 | Liga 妙谈
Open source PostgreSQL extension age for graph database was announced as the top-level project of Apache Software Foundation
删除二叉搜索树中的节点附图详解
Blue bridge: sympodial plant
【Go语言刷题篇】Go完结篇|函数、结构体、接口、错误入门学习
Self reflection of a small VC after two years of entrepreneurship
DB-Engines 2022年7月数据库排行榜:Microsoft SQL Server 大涨,Oracle 大跌
“在越南,钱就像躺在街上”
What types of Thawte wildcard SSL certificates provide
表情包坑惨职场人
随机推荐
[mathematical modeling of graduate students in Jiangxi Province in 2022] analysis and code implementation of haze removal by nucleation of water vapor supersaturation
Just today, four experts from HSBC gathered to discuss the problems of bank core system transformation, migration and reconstruction
. Net ORM framework hisql practice - Chapter 2 - using hisql to realize menu management (add, delete, modify and check)
提升复杂场景三维重建精度 | 基于PaddleSeg分割无人机遥感影像
力扣刷题日记/day1/2022.6.23
李迟2022年6月工作生活总结
一种将Tree-LSTM的强化学习用于连接顺序选择的方法
删除二叉搜索树中的节点附图详解
Numpy 的仿制 2
Unity makes revolving door, sliding door, cabinet door drawer, click the effect of automatic door opening and closing, and automatically play the sound effect (with editor extension code)
力扣刷題日記/day6/6.28
[HCIA continuous update] network management and operation and maintenance
学习路之PHP--phpstudy创建项目时“hosts文件不存在或被阻止打开”
[daily question] 556 Next bigger element III
Clever use of curl command
Reptile elementary learning
比李嘉诚还有钱的币圈大佬,刚在沙特买了楼
2022年DCMM认证全国各地补贴政策汇总
Li Kou brush question diary /day4/6.26
[system disk back to U disk] record the operation of system disk back to U disk