目錄
什么是MDL
MDL的设计目标
支持的锁类型
锁的兼容性
数据结构
首頁 資料庫 mysql教程 MySQL源码学习――MDL字典锁_MySQL

MySQL源码学习――MDL字典锁_MySQL

Jun 01, 2016 pm 01:44 PM

bitsCN.com

 

  • 什么是MDL

         MDL,Meta Data lock,元数据锁,一般称为字典锁。字典锁与数据锁相对应。字典锁是为了保护数据对象被改变,一般是一些DDL会对字典对象改变,如两个TX,TX1先查询表,然后TX2试图DROP,字典锁就会lock住TX2,知道TX1结束(提交或回滚)。数据锁是保护表中的数据,如两个TX同时更新一行时,先得到row lock的TX会先执行,后者只能等待。

  • MDL的设计目标

字典锁在设计的时候是为了数据库对象的元数据。到达以下3个目的。

1. 提供对并发访问内存中字典对象缓存(table definatin cache,TDC)的保护。这是系统的内部要求。

2. 确保DML的并发性。如TX1对表T1查询,TX2同是对表T1插入。

3. 确保一些操作的互斥性,如DML与大部分DDL(ALTER TABLE除外)的互斥性。如TX1对表T1执行插入,TX2执行DROP TABLE,这两种操作是不允许并发的,故需要将表对象保护起来,这样可以保证binlog逻辑的正确性。(貌似之前的版本存在字典锁是语句级的,导致binlog不合逻辑的bug。)

  • 支持的锁类型

        数据库理论中的基本锁类型是S、X,意向锁IS、IX是为了层次上锁而引入的。比如要修改表中的数据,可能先对表上一个表级IX锁,然后再对修改的数据上一个行级X锁,这样就可以保证其他试图修改表定义的事物因为获取不到表级的X锁而等待。

        MySQL中将字典锁的类型根据不同语句的功能,进一步细分,细分的依据是对字典的操作和对数据的操作。细分的好处是能在一定程度上提高并发效率,因为如果只定义X和S两种锁,必然导致兼容性矩阵的局限性。MySQL不遗余力的定义了如下的锁类型。

名称 意义

MDL_INTENTION_EXCLUSIVE

意向排他锁,只用于范围上锁

MDL_SHARED

共享锁,用于访问字典对象,而不访问数据。

MDL_SHARED_HIGH_PRIO

只访问字典对象(如DESC TABLE)

MDL_SHARED_READ

共享读锁,用于读取数据(如select)

MDL_SHARED_WRITE

共享写锁,用于修改数据(如update)

MDL_SHARED_NO_WRITE

共享非写锁,允许读取数据,阻塞其他TX修改数据(如alter table)

MDL_SHARED_NO_READ_WRITE

用于访问字典,读写数据

不允许其他TX读写数据

MDL_EXCLUSIVE

排他锁,可以修改字典和数据

          可以看到MySQL在ALTER TABLE的时候还是允许其他事务进行读表操作的。需要注意的是读操作的事物需要在ALTER TABLE获取MDL_SHARED_NO_WRITE锁之后,否则无法并发。这种应用场景应该是对一个较大的表进行ALTER时,其他事物仍然可以读,并发性得到了提高。

  • 锁的兼容性

          锁的兼容性就是我们经常看到的那些兼容性矩阵,X和S必然互斥,S和S兼容。MySQL根据锁的类型我们也可以知道其兼容矩阵如下:

  IX S SH SR SW SNW SNRW X IX 1 1 1 1 1 1 1 1 S 1 1 1 1 1 1 1 0 SH 1 1 1 1 1 1 1 0 SR 1 1 1 1 1 1 0 0 SW 1 1 1 1 1 0 0 0 SNW 1 1 1 1 0 0 0 0 SNRW 1 1 1 0 0 0 0 0 X 1 0 0 0 0 0 0 0

         1代表兼容,0代表不兼容。你可能发现X和IX竟然兼容,没错,其实这里的IX已经不是传统意义上的IX,这个IX是用在范围锁上,所以和X锁不互斥。

  • 数据结构

涉及到的和锁相关的数据结构主要是如下几个:

 

MDL_context:字典锁上下文。包含一个事物所有的字典锁请求。

 

MDL_request:字典锁请求。包含对某个对象的某种锁的请求。

 

MDL_ticket:字典锁排队。MDL_request就是为了获取一个ticket。

 

MDL_lock:锁资源。一个对象全局唯一。可以允许多个可以并发的事物同时获得。

 

涉及到的源码文件主要是sql/mdl.cc

 

锁资源

 

         锁资源在系统中是共享的,即全局的,存放在static MDL_map mdl_locks;的hash链表中,对于数据库中的一个对象,其hashkey必然是唯一的,对应一个锁资源。多个事务同时对一张表操作时,申请的lock也是同一个内存对象。获取mdl_locks中的lock需要通过全局互斥量保护起来mysql_mutex_lock(&m_mutex); m_mutex是MDL_map的成员。

 

上锁流程

 

        一个会话连接在实现中对应一个THD实体,一个THD对应一个MDL_CONTEXT,表示需要的mdl锁资源,一个MDL_CONTEXT中包含多个MDL_REQUEST,一个MDL_REQUEST即是对一个对象的某种类型的lock请求。每个mdl_request上有一个ticket对象,ticket中包含lock。

 

上锁的也就是根据MDL_REQUEST进行上锁。

Acquire_lock:

    if (mdl_request contains the needed ticket )

    return ticket;

    End if;

    Create a ticket;

    If (!find lock in lock_sys)

    Create a lock;

    End if

    If (lock can be granted to mdl_request)

    Set lock to ticket;

    Set ticket to mdl_request;

    Else

    Wait for lock

End if

 

 

          稍微解释下,首先是在mdl_request本身去查看有没有相等的或者stronger的ticket,如果存在,则直接使用。否则创建一个ticket,查找上锁对象对应的lock,没有则创建。检查lock是否可以被赋给本事务,如果可以直接返回,否则等待这个lock;

 

锁等待与唤醒

 

        字典对象的锁等待是发生在两个事物对同一对象上不兼容的锁导致的。当然,由于lock的唯一性,先到先得,后到的只能等待。

 

         如何判断一个lock是否可以grant给一个TX?这需要结合lock结构来看了,lock上有两个成员,grant和wait,grant代表此lock允许的事物都上了哪些锁,wait表示等待的事务需要上哪些锁。其判断一个事物是否可以grant的逻辑如下:

 

If(compatible(lock.grant, tx.locktype))

    If (compatible(lock.wait, tx.locktype))

    return can_grant;

    End if

End if

         即首先判断grant中的锁类型和当前事务是否兼容,然后判断wait中的锁类型和当前事务是否兼容。细心的话,会想到,wait中的锁类型是不需要和当前事务进行兼容性比较的,这是不是说这个比较是多余的了?其实也不是,因为wait的兼容性矩阵和上面的矩阵是不一样的,wait的兼容性矩阵感觉是在DDL等待的情况下,防止DML继续进来(wait矩阵就不写出来了,大家可以去代码里看下)。

 

比如:

 

TX1                                                      TX2                                                       TX3

 

SELECT T1

 

                                                             DROP  T1

 

                                                                                                                              SELECT T1

 

        这时候TX2会阻塞,TX3也会阻塞,被TX2阻塞,也就是说被wait的事件阻塞了,这样可能就是为了保证在DDL等待时,禁止再做DML了,因为在DDL面前,DML显得确实不是那么重要了。

 

          如何唤醒被等待的事务呢?比如唤醒TX2,当TX1结束时,会调用release_all_locks_for_name,对被锁住的事务进行唤醒,具体操作封装在reschedule_waiters函数中,重置等待时间的标记位进行唤醒,重点代码如下:

if (can_grant_lock(ticket->get_type(), ticket->get_ctx()))

    {

      if (! ticket->get_ctx()->m_wait.set_status(MDL_wait::GRANTED))

      {

        /*

          Satisfy the found request by updating lock structures.

          It is OK to do so even after waking up the waiter since any

          session which tries to get any information about the state of

          this lock has to acquire MDL_lock::m_rwlock first and thus,

          when manages to do so, already sees an updated state of the

          MDL_lock object.

        */

        m_waiting.remove_ticket(ticket);

        m_granted.add_ticket(ticket);

    }

 

 

      今天把mdl系统总体上看了一下,对锁的请求、等待以及唤醒有了初步了解。并发性的问题是最难调试的,大家如果想做锁方面的实验,可以利用VS调试中的冻结线程的功能,这样就可以确保并发情况控制完全按照你设计思路去呈现。

踏着落叶,追寻着我的梦想。转载请注明出处   作者 心中无码 bitsCN.com
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

<🎜>:泡泡膠模擬器無窮大 - 如何獲取和使用皇家鑰匙
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系統,解釋
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1664
14
CakePHP 教程
1423
52
Laravel 教程
1318
25
PHP教程
1269
29
C# 教程
1248
24
Python GIL(全域解釋器鎖定):揭秘背後的原理與效能影響 Python GIL(全域解釋器鎖定):揭秘背後的原理與效能影響 Feb 27, 2024 am 09:00 AM

pythonGIL(全域解釋器鎖)是Python中一個重要的機制,它限制了同一時刻只能有一個執行緒執行Python字節碼。這主要是為了確保Python解釋器的穩定性,因為Python的記憶體管理和垃圾回收機制都是單執行緒的。如果允許多個執行緒同時執行Python字節碼,就有可能導致記憶體損壞或其他不可預測的錯誤。 GIL的原理比較簡單。它是一個由Python解釋器維護的鎖,當一個執行緒執行Python字節碼時,它會取得GIL。其他執行緒如果想要執行Python字節碼,必須等待GIL被釋放。當GIL被釋放後,其他

如何使用Oracle 查詢表是否被鎖? 如何使用Oracle 查詢表是否被鎖? Mar 06, 2024 am 11:54 AM

標題:如何使用Oracle查詢表格是否被鎖定?在Oracle資料庫中,表鎖是指當一個事務正在對錶執行寫入操作時,其他事務想要對該表執行寫入操作或對表進行結構改變(如增加列、刪除行等)時會被阻塞。在實際開發過程中,我們經常需要查詢表格是否被鎖,以便更好地排除和處理相關問題。本文將介紹如何使用Oracle語句查詢表格是否被鎖,並給出具體的程式碼範例。要查詢表是否被鎖,我們

蘋果有鎖和無鎖的差別是什麼 詳細介紹:iphone有鎖和無鎖的差別對比 蘋果有鎖和無鎖的差別是什麼 詳細介紹:iphone有鎖和無鎖的差別對比 Mar 28, 2024 pm 03:10 PM

蘋果手機是最近大家選擇最廣泛的一款手機,但我們常常在網路上看到大家在討論蘋果手機有鎖與無鎖的差別,會糾結於要購買哪一種。今天陳斯琪就為大家普及一下iphone有鎖和無鎖的差別,為大家排憂解難。其實二者在外觀、功能上並沒有太大差別,關鍵就在價格以及使用上的一些問題。什麼是有鎖版與無鎖版沒有鎖版限制的蘋果手機指不受業者的限制,任何一家業者的SIM卡都可以正常使用。有鎖版是指加了網路鎖,只能使用指定的電信業者提供的SIM卡,不能使用其他的。其實沒有鎖版的蘋果手機就像我們的全網通手機一樣可以使用移動、

Redis實現分散式鎖定的效能對比 Redis實現分散式鎖定的效能對比 Jun 20, 2023 pm 05:46 PM

隨著網路應用的規模越來越大,分散式系統也越來越常見。在這些系統中,分散式鎖是一項不可或缺的功能。由於分散式鎖需求旺盛,因此存在著各種各樣的實現方式。其中,Redis是一種流行的,在分散式鎖定實作中被廣泛應用的工具。在本文中,我們將探討Redis實現分散式鎖的效能比較。一、Redis基礎概念在討論Redis的分散式鎖定效能之前,我們需要先了解一些Redis的基礎概

golang函式中的鎖是如何實現的? golang函式中的鎖是如何實現的? Jun 05, 2024 pm 12:39 PM

Go語言中的鎖實作同步並發程式碼,防止資料競爭:Mutex:互斥鎖,保證同一時間只有一個goroutine取得鎖,用於臨界區控制。 RWMutex:讀寫鎖,允許多個goroutine同時讀取數據,但僅一個goroutine同時寫入數據,適用於需要頻繁讀寫共享數據的場景。

Python 並發程式設計中的鎖與同步:保持你的程式碼安全可靠 Python 並發程式設計中的鎖與同步:保持你的程式碼安全可靠 Feb 19, 2024 pm 02:30 PM

並發程式設計中的鎖定與同步在並發程式設計中,多個行程或執行緒同時執行,這可能會導致資源爭用和不一致性問題。為了解決這些問題,需要使用鎖定和同步機制來協調對共享資源的存取。鎖的概念鎖是一種機制,它允許一次只有一個執行緒或程序存取共享資源。當一個執行緒或程序獲得鎖時,其他執行緒或程序將被阻止存取該資源,直到鎖被釋放。鎖的類型python中有幾種類型的鎖:互斥鎖(Mutex):確保一次只有一個執行緒或程序可以存取資源。條件變數:允許執行緒或行程等待某個條件,然後取得鎖定。讀寫鎖:允許多個執行緒同時讀取資源,但只允許一個執行緒寫入資

深入解析Golang鎖的底層實作機制 深入解析Golang鎖的底層實作機制 Dec 28, 2023 am 11:26 AM

Golang鎖的底層實作原理詳解,需要具體程式碼範例概述:並發程式設計是現代軟體開發中非常重要的一部分,而鎖是實現並發控制的機制。在Golang中,鎖的概念被廣泛應用於並發程式設計。本篇文章將深入探討Golang鎖的底層實作原理,並提供具體的程式碼範例。互斥鎖(Mutex)的底層實作原理互斥鎖是Golang中最常用的鎖類型之一。它採用了一個底層資料結構sync.M

如何在Go語言中使用鎖實現線程安全 如何在Go語言中使用鎖實現線程安全 Mar 23, 2024 pm 07:00 PM

在Go語言中使用鎖實現線程安全隨著並發程式設計的不斷普及,確保資料在多個goroutine之間安全存取變得尤為重要。在Go語言中,可以使用鎖定來實現執行緒安全,確保共享資源在並發環境下的存取不會導致資料競爭問題。本文將詳細介紹如何在Go語言中使用鎖實現線程安全,同時提供具體的程式碼範例。什麼是鎖鎖是一種並發程式設計中常用的同步機制,可以在多個goroutine之間協調對共

See all articles