MySQL高級9-鎖

来源:https://www.cnblogs.com/Se7eN-HOU/archive/2023/09/04/17641868.html
-Advertisement-
Play Games

一、簡介 鎖是電腦協調多個進程或線程併發訪問某一資源的機制。在資料庫中,除了傳統的計算資源(CPU、RAM、i/O)的掙用以外,數據也是一種供許多用戶共用的資源。如何保證數據併發訪問的一致性,有效性是所有資料庫必須解決的一個問題,鎖衝突也是影響資料庫併發訪問性能的一個重要因素,從這個角度來說,鎖對 ...


一、簡介

  鎖是電腦協調多個進程或線程併發訪問某一資源的機制。在資料庫中,除了傳統的計算資源(CPU、RAM、i/O)的掙用以外,數據也是一種供許多用戶共用的資源。如何保證數據併發訪問的一致性,有效性是所有資料庫必須解決的一個問題,鎖衝突也是影響資料庫併發訪問性能的一個重要因素,從這個角度來說,鎖對資料庫而言顯得尤其重要,也更加複雜。

 

二、分類

  MySQL中的鎖、按照鎖的粒度分,分為以下三類

  • 全局鎖:鎖定數據量的所有表
  • 表級鎖:每次操作鎖住整張表
  • 行級鎖:每次操作鎖住對應的行數據

 

三、全局鎖

  3.1 簡介

    全局鎖就是對整個資料庫實例枷鎖,加鎖後整個實例就處於只讀狀態,後續的DML的寫語句,DDL語句,以及更新操作的事物提交語句都會被阻塞,其典型的使用場景是做全庫的邏輯備份,對所有的表進行鎖定,從而獲取一致性視圖,保證數據的完整性。

  3.2 添加全局鎖語法 

flush tables with read lock;

  3.3 打開全局鎖語法

unlock tables;

  3.4 案例一

  

    說明1:客戶端1中設置了全局鎖

    說明2:客戶端2中,執行查詢語句正常的,但是執行DML語句中的更新操作卻是處於阻塞狀態

  3.5 案例二

  

    說明1:客戶端1中設置了全局鎖

    說明2:客戶端3中做了資料庫備份的語句,其中mysqldump是和mysql一樣由MySQL伺服器提供的資料庫備份的命令

    說明3:當資料庫設置了全局鎖的時候,不影響資料庫的備份

    說明4:MySQL備份實在終端命令行模式下,不是在資料庫命令模式下,註意!註意!註意!

  3.6 案例三

  

    說明1:客戶端1釋放全局鎖 

    說明2:客戶端2的更新語句馬上執行成功

    說明3:通過客戶端2的更新語句的執行時間37min,可以說明該語句被客戶端1的全局鎖,阻塞了37分鐘

  3.7 全局鎖特點

    資料庫中加全局鎖,是一個比較重的操作,存在以下問題

    • 如果在主庫上備份,那麼在備份期間都不能執行更新,業務上基本就是停擺狀態
    • 如果在從庫上備份,那麼在備份期間從庫不能執行主庫同步過來的二進位日誌(binlog),會導致主從延遲

    在innoDB引擎中,我們可以在備份時加上 --single-transaction 參數來完成不加鎖的一致性數據備份

 

四、表級鎖

  4.1 簡介

    表級鎖,每次操作鎖住整張表,鎖定粒度大,發生鎖衝突的概率最高,併發度最低,應用在MyISAM,InnoDB等存儲引擎中

  4.2 表級鎖的分類

    • 表鎖
    • 元數據鎖(meta data lock, MDL)
    • 意向鎖

五、表鎖

  5.1 表鎖的分類

    • 表共用讀鎖(read lock)
    • 表獨占寫鎖(write lock)

  5.2 表鎖語法

    加鎖:lock tables 表名... read/write

    釋放鎖:unlock tables / 客戶端斷開連接

    說明1:加鎖的時候,可以多張表同時加鎖

    說明2:客戶端斷開連接也能釋放表鎖

  5.3 讀鎖案例

      

   

      說明1:當對一個表添加讀鎖,不會影響其讀取數據,但是會影響其新增,修改,刪除的操作語句

    說明2:當對一個表添加讀鎖,不會影響其他客戶端讀取數據,但是會讓其他客戶端的新增,修改,刪除等操作語句處於阻塞狀態。

    說明3:當把鎖釋放了,自己及其他客戶端的新增,刪除和修改語句才會結束阻塞。

  5.4 寫鎖案例

    

    

    說明1:添加寫鎖的客戶端可以正常對錶進行查詢和增刪改等操作

    說明2:其他客戶端的查詢,修改,新增,刪除都操作在有寫鎖的情況下,都要處於阻塞狀態,直到其添加寫鎖的客戶端釋放寫鎖。

    說明3:上圖中的修改等應該是DML語句(Data Manipulation Language 數據操作語言,用來對資料庫表中的數據進行增刪改的),不是DDL語句(DDL: Data Definition Language 數據定義語言,用來定義資料庫對象,資料庫,表,欄位)

 

六、元數據鎖

  6.1 元數據鎖簡介:

    元數據鎖(meta data lock):簡稱 MDL

    MDL加鎖過程是系統自動控制,無需顯示使用,在訪問一張表的時候會自動加上。MDL鎖主要作用是維護表元數據的數據一致性,在表上有活動事務的時候,不可以對元數據進行寫入操作,為了避免DML和DDL衝突,保證讀寫的正確性

  6.2 元數據鎖的類型

    在MySQL5.5 中引入了MDL, 當對一張表進行增刪改查的時候,加MDL讀鎖(共用);當對錶結構進行變更操作的時候,加MDL寫鎖(排他)

    

  6.3 案例

    

    說明1:select 和 update 分別會添加SHARED_READ 和 SHARED_WRITE, 但是SHARED_READ 和 SHARED_WRITE是相容的,所以其他客戶端是可以做更刪改查的

    

     說明2:alter table 會產生EXCLUSIVE 鎖。該鎖與其他MDL都互斥。ß

  6.4 查看元數據鎖

select object_type,object_schema,object_name,lock_type,locak_duration from  performance_schema.metadata_locks;

    

 

七、意向鎖

  7.1 意向鎖介紹

    為了避免DML在執行時,加的行鎖與表鎖的衝突,在InnoDB中引入了意向鎖,使得表鎖不用檢查每一行數據是否加鎖,使用意向鎖來減少表鎖的檢查。

    情況1:不加意向鎖

   

    說明1:線程A 開啟一個事務,並且使用DML語句更新數據,此時會對更新的數據添加行鎖,

    說明2:此時線程A 的事務並沒有提交事務,線程B又對該表添加一個表鎖,此時添加表鎖的時候,就會從第一條數據,依次檢查到最後一條數據,看是否有其他的鎖,如果有其他鎖,需要等到其他鎖釋放了,才能執行添加表鎖

    說明3:此時添加表鎖的效率就非常低  

    情況2:加意向鎖

   

    說明4:當線程A 開啟了一個事務,並且執行了DML的更新數據的語句,此時除了會給該行數據添加行鎖之外,還會添加一個基於整表的意向鎖。

    說明5:當線程B 再次添加表鎖的時候,就不用逐行排查是否有行鎖了,而是直接檢查整表是否有意向鎖,如果意向鎖和表鎖相容則直接加表鎖

    說明6:如果表鎖和意向鎖不相容,則仍然會等到意向鎖釋放了,表鎖才能添加成功。

  7.2 意向鎖的類型

    • 意向共用鎖(IS): 由語句 select ... lock in share mode 添加。與表鎖共用鎖(read)相容,與表鎖排它鎖(write)互斥。
    • 意向排它鎖(IX): 由 insert, update, delete, select ... for update 添加。  與表鎖共用鎖(read)及排它鎖(write)都互斥,意向鎖之間不會互斥

  7.3 意向鎖查詢

select object_schema, object_name, index_name, lock_type, lock_mode, lock_data from performance_schema.data_locks;

  7.4 意向鎖案例

    

    說明1:客戶端1開啟事務,並且查詢語句時設置一個意向共用鎖(IS)

    說明2:客戶端2在客戶端1的事務未提交的時候,創建read鎖成功

    說明3:意向共用鎖(IS)和read共用鎖相容

    

    說明4:執行update等DML語句時,MySQL會自動添加行鎖和排他意向鎖。

    

    說明5:排他意向鎖與共用read鎖和排他write鎖都互斥

    說明6:意向鎖的作用主要是解決行鎖和表鎖之間的矛盾。

 

八、行級鎖

  8.1、簡介

    行級鎖,每次操作鎖住對應的行數據,所得粒度最小,發生鎖衝突的概率最低,併發度最高,應用在InnoDB存儲引擎中

    InnoDB的數據是基於索引組織的行鎖是通過對索引上的索引項加鎖來實現的,而不是對記錄加的鎖,對於行級鎖,主要分為一下三類

    1、行鎖(record lock): 鎖定單個行記錄的鎖,防止其他事務對此行進行update和delete,在read committed和repeatable read隔離級別下都支持。

    

     2、間隙鎖(Gap lock): 鎖定索引記錄間隙(不包含記錄),確保索引記錄間隙不變,防止其他事務在這個間隙進行insert,產生幻讀,在repeatable read隔離級別下支持

     

     3、臨建鎖(Next-Key lock): 行鎖和間隙鎖組合,同時鎖住數據,並鎖住數據前面的間隙,在repeatable read隔離級別下支持

     

   8.2 行鎖

    InnoDB 實現了一下兩種類型的行鎖

    1、共用鎖(S): 允許一個事物去讀一行,阻止其他事務獲得相同數據集的排他鎖。

    2、排他鎖(X): 允許獲取排他鎖的事務更新數據,阻止其他事物獲得相同數據集的共用鎖和排他鎖

    

     3、不同語句的枷鎖情況

    

     4、預設情況下,InnoDB在 repeatable read隔離級別下運行,InnoDB使用next-key鎖進行行搜索和索引掃描,以防止幻讀

    • 針對唯一索引進行檢索時,對已經存在的記錄進行等值匹配時,將會自動優化為行鎖
    • InnoDB的行鎖是針對於索引加的鎖,不通過索引檢索數據,那麼InnoDB將對錶中的所有記錄加鎖,此時就會升級為表鎖

   8.3 行鎖案例

    

    說明1:客戶端1開始事務,並執行查詢語句,客戶端2鎖的情況,顯示為空表

    說明2:客戶端執行查詢語句,並手動添加共用,在客戶端2中,查詢lock_mode欄位中有一個S,即共用鎖: 

    

    說明3:客戶端2也手動開啟了一個共用鎖,並且成功,說明共用鎖與共用鎖之間是相容的。

    

    說明5:在客戶端2上執行更新id=2的數據成功,並且查詢可以看出,自動增加了一個(X)排它鎖。

    說明6:  因為更新id=2的數據時,id=2的這一行上沒有其他的鎖,所以可以執行成功

    說明7:因為在客戶點1上已經對id=1的行上添加了一個共用鎖(S), 這是客戶端2對id=1的數據做update操作時會自動在id=1的數據上在添加一個排它鎖(X),這是id=1的數據上就會有共用鎖(S)和排它鎖(X),又因為共用鎖和排它鎖不相容,所以在update id = 1 的時候就會阻塞,需要等到共用鎖(S)釋放了,才能執行成功。

    

    說明8:排它鎖X 與 排它鎖X 也是不相容的。

    

     說明9:InnoDB的行鎖是針對於索引加的鎖,不通過索引檢索數據,那麼InnoDB將對錶中的所有記錄加鎖,此時就會升級為表鎖

    

     說明10:根據業務需要,儘量使用索引當所查詢條件,既快又減少阻塞。

 

九、間隙鎖/臨建鎖

  9.1 預設情況下、InnoDB在 repeatable read 事務隔離級別運行,InnoDB使用next-key 鎖進行搜索和索引掃描,以防止幻讀。

    

  9.2 索引上的等值查詢(唯一索引),給不存在的記錄加鎖時,優化為間隙鎖

    

  9.3 索引上的等值查詢(普通索引),向右遍歷時最後一個值不滿足查詢需求時,next-key lock 退化為間隙鎖

    

    說明1:非唯一索引,即普通索引,在做等值查詢的時候,會添加三把鎖

    說明2:在查詢的當條記錄上添加一個共用鎖,允許其他事務繼續查詢該記錄

    說明3:通過標註6的位置,lock_type 為S,REC_NOT_GAP 說明同時給這一行添加一個行鎖

    說明4:通過標註7的位置,lock_type 為 S,GAP 說明在第一個不滿足查詢數據的前面也會加一個間隙鎖。其目的是為了防止在select 查詢的時候,其他事務去往這個索引之前插入或者修改數據,這樣查詢就會出現幻讀的現象。

  9.4 索引上的範圍查詢(唯一索引),會訪問到不滿足條件的第一個值為止。

    

  註意:間隙鎖唯一的目的是防止其他事務插入間隙,間隙鎖可以共存,一個事務採用的間隙鎖不會阻止另一個事物在同一間隙上採用間隙鎖 


侯哥語錄:我曾經是一個職業教育者,現在是一個自由開發者。我希望我的分享可以和更多人一起進步。分享一段我喜歡的話給大家:"我所理解的自由不是想乾什麼就乾什麼,而是想不幹什麼就不幹什麼。當你還沒有能力說不得時候,就努力讓自己變得強大,擁有說不得權利。"
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • ## 1、線程的基本管控 包含頭文件``後,通過構建`std::thread`對象啟動線程,任何可調用類型都適用於`std::thread`。 ```c++ void do_some_work(); struct BackgroundTask { void operator()() const; } ...
  • > 本文深入探討了Go語言的多個關鍵方面,從其簡潔的語法、強大的併發支持到出色的性能優勢,進一步解析了Go在雲原生領域的顯著應用和廣泛的跨平臺支持。文章結構嚴謹,逐一分析了Go語言在現代軟體開發中所占據的重要地位和其背後的技術原理。 > 關註TechLeadCloud,分享互聯網架構、雲服務技術的全 ...
  • java中要實現excel新老格式的轉換比較麻煩,開源庫也沒幾個好用的。用ChatGpt查詢也是推薦直接用POI,下麵是藉助ChatGPT寫出來的代碼,經過小小修改,格式轉換良好,基本能用,就是效率比較低下。將就著用吧,哎! /** * Excel格式從xls轉換成xlsx格式 * * @param ...
  • Hi,大家好,我是一路狂奔的烏龜 很高興遇見你~ 我們可能會在睡前下載文件,而不想自己的電腦一整晚都在運行,浪費電也損耗電腦硬體 基於這個日常需求(你怎麼知道我喜歡在半夜下載文件?) 所以我就開發了這款軟體 至今已經修改了三四個版本了,基本滿足要求了就不再更新了 好了, 接下來,,就容我介紹一下這款 ...
  • # iptables、共用上網SNAT、埠轉發DNAT ## 1.防火牆概述 封端⼝,封ip 實現NAT功能 共用上⽹ 端⼝映射(端⼝轉發),ip映射 ## 2.防火牆 ### 2.1防火牆種類以及使用說明 硬體:整個企業入口 軟體:開源軟體 網站內部 封ip iptables 雲防火牆 安全組 ...
  • 提要:本系列文章主要參考`MIT 6.828課程`以及兩本書籍`《深入理解Linux內核》` `《深入Linux內核架構》`對Linux內核內容進行總結。 記憶體管理的實現覆蓋了多個領域: 1. 記憶體中的物理記憶體頁的管理 2. 分配大塊記憶體的伙伴系統 3. 分配較小記憶體的slab、slub、slob分 ...
  • # Keepalived高可用集群 ## 高可用集群簡介 **什麼是高可用集群?** 高可用集群 (High Availability;Cluster,簡稱HA Cluster) ,是指以減少服務中斷時間為目的的伺服器集群技術。它通過保護用戶的業務程式對外不間斷提供的服務,把因軟體、硬體、人為造成的 ...
  • ![](https://img2023.cnblogs.com/blog/3076680/202309/3076680-20230902230646018-938272092.png) # 1. 元數據 ## 1.1. metadata ## 1.2. 關於數據的數據 ## 1.3. 數據字典 ## ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...