在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

MySQL自增主鍵一定是連續的嗎?

dyquk4xk2p3d ? 來源:CSDN ? 2023-06-11 11:35 ? 次閱讀

測試環境:

MySQL版本:8.0

數據庫表:T (主鍵id,唯一索引c,普通字段d)

27e06496-05e9-11ee-962d-dac502259ad0.png

如果你的業務設計依賴于自增主鍵的連續性,這個設計假設自增主鍵是連續的。但實際上,這樣的假設是錯的,因為自增主鍵不能保證連續遞增。

一、自增值的屬性特征:

1. 自增主鍵值是存儲在哪的?

MySQL5.7版本

在 MySQL 5.7 及之前的版本,自增值保存在內存里,并沒有持久化。每次重啟后,第一次打開表的時候,都會去找自增值的最大值max(id),然后將max(id)+1作為這個表當前的自增值。

MySQL8.0之后版本

在 MySQL 8.0 版本,將自增值的變更記錄在了redo log中,重啟的時候依靠redo log恢復重啟之前的值。

可以通過看表詳情查看當前自增值,以及查看表參數詳情AUTO_INCREMENT值(AUTO_INCREMENT就是當前數據表的自增值)

27eca3dc-05e9-11ee-962d-dac502259ad0.png

2. 自增主鍵值的修改機制?

在表t中,我定義了主鍵id為自增值,在插入一行數據的時候,自增值的行為如下:

如果插入數據時 id 字段指定為 0、null 或未指定值,那么就把這個表當前的AUTO_INCREMENT值填到自增字段;

如果插入數據時 id 字段指定了具體的值,就直接使用語句里指定的值。

根據要插入的值和當前自增值的大小關系,自增值的變更結果也會有所不同。假設,某次要插入的值是 X,當前的自增值是 Y。

如果 X

如果 X≥Y,就需要把當前自增值修改為新的自增值。

二、新增語句自增主鍵是如何變化的:

我們執行以下SQL語句,來觀察自增主鍵是如何進行變化的

insertintotvalues(null,1,1);

流程圖如下所示

27fc6628-05e9-11ee-962d-dac502259ad0.png

流程步驟:

AUTO_INCREMENT=1(表示下一次插入數據時,如果需要自動生成自增值,會生成 id=1。)

insert into t values(null, 1, 1)(執行器調用 InnoDB 引擎接口寫入一行,傳入的這一行的值是 (0,1,1))

get AUTO_INCREMENT=1(InnoDB 發現用戶沒有指定自增 id 的值,獲取表 t 當前的自增值 1 )

AUTO_INCREMENT=2 insert into t values(1, 1, 1)(將傳入的行的值改成 (1,1,1),并把自增值改為2)

insert (1,1,1)執行插入操作,至此流程結束

大家可以發現,在這個流程當中是先進行自增值的+1,在進行新增語句的執行的。大家可以發現這個操作并沒有進行原子操作,如果SQL語句執行失敗,那么自增是不是就不會連續了呢?

三、自增主鍵值不連續情況:(唯一主鍵沖突)

當我執行以下SQL語句時

insertintotvalues(null,1,1);

第一次我們可以進行新增成功,根據自增值的修改機制。如果插入數據時 id 字段指定為 0、null 或未指定值,那么就把這個表當前的AUTO_INCREMENT值填到自增字段;

當我們第二次在執行以下SQL語句時,就會出現錯誤。因為我們表中c字段是唯一索引,會出現Duplicate key error錯誤導致新增失敗。

280643aa-05e9-11ee-962d-dac502259ad0.png

例如:

AUTO_INCREMENT=2(表示下一次插入數據時,如果需要自動生成自增值,會生成 id=2。)

insert into t values(null, 1, 1)(執行器調用 InnoDB 引擎接口寫入一行,傳入的這一行的值是 (0,1,1))

get AUTO_INCREMENT=2(InnoDB 發現用戶沒有指定自增 id 的值,獲取表 t 當前的自增值 2 )

AUTO_INCREMENT=3 insert into t values(2, 1, 1)(將傳入的行的值改成 (2,1,1),并把自增值改為3)

insert (2,1,1)執行插入操作,由于已經存在 c=1 的記錄,所以報Duplicate key error,語句返回。

可以看到,這個表的自增值改成 3,是在真正執行插入數據的操作之前。這個語句真正執行的時候,因為碰到唯一鍵 c 沖突,所以 id=2 這一行并沒有插入成功,但也沒有將自增值再改回去。所以,在這之后,再插入新的數據行時,拿到的自增 id 就是 3。也就是說,出現了自增主鍵不連續的情況。

四、自增主鍵值不連續情況:(事務回滾)

其實事務回滾原理也和上面一樣,都是因為異常導致新增失敗,但是自增值沒有進行回退。

五、自增主鍵值不連續情況:(批量插入)

批量插入數據的語句,MySQL 有一個批量申請自增 id 的策略:

語句執行過程中,第一次申請自增 id,會分配 1 個;

1 個用完以后,這個語句第二次申請自增 id,會分配 2 個;

2 個用完以后,還是這個語句, 第三次申請自增 id,會分配 4 個;

依此類推,同一個語句去申請自增 id,每次申請到的自增 id 個數都是上一次的兩倍。

執行以下SQL語句(在表t中先新增了4條數據,在創建表tt把表t數據進行批量新增)

insertintotvalues(null,1,1);
insertintotvalues(null,2,2);
insertintotvalues(null,3,3);
insertintotvalues(null,4,4);
createtablettliket;
insertintott(c,d)selectc,dfromt;

insertintottvalues(null,5,5);

第一次申請到了 id=1,第二次被分配了 id=2 和 id=3, 第三次被分配到 id=4 到 id=7。當我們再執行insert into t2 values(null, 5,5),實際上插入的數據就是(8,5,5),出現了自增主鍵不連續的情況。

2812d1e2-05e9-11ee-962d-dac502259ad0.png

六、自增主鍵值的優化

1.什么是自增鎖

自增鎖是一種比擬非凡的表級鎖。并且在事務向蘊含了AUTO_INCREMENT列的表中新增數據時就會去持有自增鎖,假如事務 A 正在做這個操作,如果另一個事務 B 嘗試執行 INSERT語句,事務 B 會被阻塞住,直到事務 A 開釋自增鎖。

2.自增鎖有哪些優化

在 MySQL 5.0 版本的時候,自增鎖的范圍是語句級別。也就是說,如果一個語句申請了一個表自增鎖,這個鎖會等語句執行結束以后才釋放。顯然,這樣設計會影響并發度。在MySQL 5.1.22 版本引入了一個新策略,新增參數innodb_autoinc_lock_mode,默認值是 1。

傳統模式(Traditional)

這個參數的值被設置為 0 時,表示采用之前 MySQL 5.0 版本的策略,即語句執行結束后才釋放鎖;

傳統模式他可以保證數據一致性,但是如果有多個事務并發的執行 INSERT 操作,AUTO-INC的存在會使得 MySQL 的性能略有降落,因為同時只能執行一條 INSERT 語句。

間斷模式(Consecutive)

這個參數的值被設置為 1 時:普通 insert 語句,自增鎖在申請之后就馬上釋放;類似insert … select這樣的批量插入數據的語句,自增鎖還是要等語句結束后才被釋放;

間斷模式他可以保證數據一致性,但是如果有多個事務并發的執行 INSERT 批量操作時,就會進行鎖等待狀態。如果我們業務插入數據量很大時,這個時候MySQL的性能就會大大下降。

穿插模式(Interleaved)

這個參數的值被設置為 2 時,所有的申請自增主鍵的動作都是申請后就釋放鎖。

穿插模式他沒有進行任何的上鎖設置。在一定情況下是保證了MySQL的性能,但是他無法保證數據的一致性。如果我們在穿插模式下進行主從復制時,如果你的binlog格式不是row格式,主從復制就會出現不一致。

七、MySQL8.0做了哪些優化

在MySQL8.0之后版本,已經默認設置為innodb_autoinc_lock_mode=2,binlog_format=row.。這樣更有利與我們在insert … select這種批量插入數據的場景時,既能提升并發性,又不會出現數據一致性問題。





審核編輯:劉清

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • SQL
    SQL
    +關注

    關注

    1

    文章

    766

    瀏覽量

    44169
  • MYSQL數據庫
    +關注

    關注

    0

    文章

    96

    瀏覽量

    9410

原文標題:被問懵了:MySQL 自增主鍵一定是連續的嗎?

文章出處:【微信號:良許Linux,微信公眾號:良許Linux】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    怎么簡單實現由Labview讀取的串口數據寫入mysql5.7數據庫中?

    怎么簡單實現由Labview讀取的串口數據寫入mysql5.7數據庫中? 已實現:串口數據的接收處理 mysql5.7的安裝(已測試數據庫正常運行) 愿付費解決此問題(QQ:8
    發表于 01-11 22:05

    #硬聲創作季 【MySQL調優】為什么推薦使用整型的主鍵而不是UUID

    數據庫MySQL
    Mr_haohao
    發布于 :2022年09月14日 07:41:14

    阿里云mysql數據庫怎么設置主鍵和時間格式怎么顯示時分秒?

    `需要將測試的數據保存到阿里云mysql數據庫上,利用NI的數據庫工具包怎么創建表實現主鍵?還有保存的時間數據只顯示日期,不顯示時分秒(用DB Tools Format Datet
    發表于 11-13 10:39

    Python常用運算寫法

    在學習任何種編程語言,運算方法的學習是不可避免的,其中運算符++的使用也是其中的重點,很多人容易弄不明白其用法,尤其是在Python中,其用法更加的不同!我們可以寫個實例來說明
    發表于 04-09 17:45

    labview向oracle插入數據,怎樣可以主鍵1?如果不插入主鍵的字段,會報插入的數目與表中的數據不相等

    本帖最后由 電子人steve 于 2018-5-23 20:30 編輯 labview向oracle數據庫插入數據時,怎樣可以主鍵1啊,服務器數據庫表中設置了
    發表于 05-23 18:53

    MySQL表分區類型及介紹

    表分區是將個表的數據按照一定規則水平劃分成不同的邏輯塊,并分別進行物理存儲,這個規則就叫做分區函數,可以有不同的分區規則。通過show plugins語句查看當前MySQL是否支持表分區功能
    發表于 06-29 16:31

    關于MySQL的基礎知識簡析

    系統環境MAC OS 10.10MySQL版本,5.7.9Mac 安裝mysql很簡單,官網下載安裝包,雙擊安裝就可以了,有幾個地方需要注意在Mac下用DMG包新安裝mysql,在安裝完畢最后
    發表于 11-03 11:50

    21個MySQL表設計的經驗準則

    主鍵設計的話,最好不要與業務邏輯有所關聯。有些業務上的字段,比如身份證,雖然是唯的,些開發者喜歡用它來做主鍵,但是不是很建議哈。主鍵最好
    的頭像 發表于 01-12 10:07 ?582次閱讀

    MySQL主鍵一定是連續的嗎?

    眾所周知,主鍵可以讓聚集索引盡量地保持遞增順序插入,避免了隨機查詢,從而提高了查詢效率
    的頭像 發表于 02-20 18:06 ?748次閱讀

    MySQL主鍵一定是連續的嗎?

    如果你的業務設計依賴于主鍵連續性,這個設計假設主鍵
    的頭像 發表于 03-21 16:55 ?643次閱讀

    主鍵不用隨機字符串用什么?主鍵

    主鍵不用隨機字符串用什么?主鍵主鍵就是最佳
    的頭像 發表于 05-09 09:04 ?646次閱讀
    <b class='flag-5'>主鍵</b>不用隨機字符串用什么?<b class='flag-5'>主鍵</b><b class='flag-5'>自</b><b class='flag-5'>增</b>?

    線上MySQLid用盡怎么辦?

    MySQLid都定義了初始值,然后不斷加步長。雖然自然數沒有上限,但定義了表示這個數的字節長度,計算機存儲就有上限。
    的頭像 發表于 05-22 10:23 ?538次閱讀
    線上<b class='flag-5'>MySQL</b>的<b class='flag-5'>自</b><b class='flag-5'>增</b>id用盡怎么辦?

    id的機制不同在mysql的索引結構以及優缺點

    的時候,mysql官方推薦不要使用uuid或者不連續不重復的雪花id(long形且唯,單機遞增),而是推薦連續
    的頭像 發表于 06-30 10:19 ?815次閱讀
    id的機制不同在<b class='flag-5'>mysql</b>的索引結構以及優缺點

    MySQL索引的常用知識點

    索引結構:B+樹 索引其實是種數據結構 注意B+樹是MySQL,索引默認的結構;張表至少有個索引(主鍵索引),是可以有多個索引的
    的頭像 發表于 09-30 16:43 ?473次閱讀

    主鍵去哪了?---次開發過程中的思考

    TABLE `example_table` ( `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT COMMENT '主鍵
    的頭像 發表于 09-05 14:12 ?270次閱讀
    <b class='flag-5'>自</b><b class='flag-5'>增</b><b class='flag-5'>主鍵</b>去哪了?---<b class='flag-5'>一</b>次開發過程中的思考
    主站蜘蛛池模板: 东方天堂网| 黄色免费大全| 午夜在线观看完整高清免费| 国产成人精品系列在线观看| 国产欧美另类第一页| 99久久成人国产精品免费 | 又黄又爽又猛午夜性色播在线播放| 国产精品天天干| 女bbbbxxxx毛片视频| 婷婷丁香五| 三级五月天| 青青草99热久久| 亚洲国产人久久久成人精品网站 | 二级黄绝大片中国免费视频| 7777在线| 人人爱天天操| 在线黄色网| 久久15| 狠狠干狠狠插| 欧美性精品| 九九热免费在线观看| 亚洲无线视频| 日韩1页| 四虎国产在线| 天堂网最新版中文| 日本网站免费| 日本特级淫片免费看| 久碰香蕉精品视频在线观看| 俺来也婷婷| 国产在线播| 日本aaaa| 色女人综合| 亚洲高清毛片| 成片一卡三卡四卡免费网站| 午夜啪啪福利视频| 毛片网站免费| 日本午夜三级| 国产在线播| 神马午夜嘿嘿嘿| 91大神免费视频| 亚洲成熟|