mysql表鎖怎么鎖 mysql鎖表語句

MySQL從入門到精通(九) MySQL鎖,各種鎖

鎖是計算機協(xié)調(diào)多個進(jìn)程或線程并發(fā)訪問某一資源的機制,在數(shù)據(jù)庫中,除傳統(tǒng)的計算資源(CPU、RAM、I/O)爭用外,數(shù)據(jù)也是一種供許多用戶共享的資源,如何保證數(shù)據(jù)并發(fā)訪問的一致性,有效性是所有數(shù)據(jù)庫必須解決的一個問題,鎖沖突也是影響數(shù)據(jù)庫并發(fā)訪問性能的一個重要因素,從這個角度來說,鎖對數(shù)據(jù)庫而言是尤其重要,也更加復(fù)雜。MySQL中的鎖,按照鎖的粒度分為:1、全局鎖,就鎖定數(shù)據(jù)庫中的所有表。2、表級鎖,每次操作鎖住整張表。3、行級鎖,每次操作鎖住對應(yīng)的行數(shù)據(jù)。

目前創(chuàng)新互聯(lián)公司已為上千的企業(yè)提供了網(wǎng)站建設(shè)、域名、虛擬空間、網(wǎng)站改版維護、企業(yè)網(wǎng)站設(shè)計、泉山網(wǎng)站維護等服務(wù),公司將堅持客戶導(dǎo)向、應(yīng)用為本的策略,正道將秉承"和諧、參與、激情"的文化,與客戶和合作伙伴齊心協(xié)力一起成長,共同發(fā)展。

全局鎖就是對整個數(shù)據(jù)庫實例加鎖,加鎖后整個實例就處于只讀狀態(tài),后續(xù)的DML的寫語句,DDL語句,已經(jīng)更新操作的事務(wù)提交語句都將阻塞。其典型的使用場景就是做全庫的邏輯備份,對所有的表進(jìn)行鎖定,從而獲取一致性視圖,保證數(shù)據(jù)的完整性。但是對數(shù)據(jù)庫加全局鎖是有弊端的,如在主庫上備份,那么在備份期間都不能執(zhí)行更新,業(yè)務(wù)會受影響,第二如果是在從庫上備份,那么在備份期間從庫不能執(zhí)行主庫同步過來的二進(jìn)制日志,會導(dǎo)致主從延遲。

解決辦法是在innodb引擎中,備份時加上--single-transaction參數(shù)來完成不加鎖的一致性數(shù)據(jù)備份。

添加全局鎖: flush tables with read lock; 解鎖 unlock tables。

表級鎖,每次操作會鎖住整張表.鎖定粒度大,發(fā)送鎖沖突的概率最高,并發(fā)讀最低,應(yīng)用在myisam、innodb、BOB等存儲引擎中。表級鎖分為: 表鎖、元數(shù)據(jù)鎖(meta data lock, MDL)和意向鎖。

表鎖又分為: 表共享讀鎖 read lock、表獨占寫鎖write lock

語法: 1、加鎖 lock tables 表名 ... read/write

2、釋放鎖 unlock tables 或者關(guān)閉客戶端連接

注意: 讀鎖不會阻塞其它客戶端的讀,但是會阻塞其它客戶端的寫,寫鎖既會阻塞其它客戶端的讀,又會阻塞其它客戶端的寫。大家可以拿一張表來測試看看。

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

在MySQL5.5中引入了MDL,當(dāng)對一張表進(jìn)行增刪改查的時候,加MDL讀鎖(共享);當(dāng)對表結(jié)構(gòu)進(jìn)行變更操作時,加MDL寫鎖(排他).

查看元數(shù)據(jù)鎖:

select object_type,object_schema,object_name,lock_type,lock_duration from performance_schema_metadata_locks;

意向鎖,為了避免DML在執(zhí)行時,加的行鎖與表鎖的沖突,在innodb中引入了意向鎖,使得表鎖不用檢查每行數(shù)據(jù)是否加鎖,使用意向鎖來減少表鎖的檢查。意向鎖分為,意向共享鎖is由語句select ... lock in share mode添加。意向排他鎖ix,由insert,update,delete,select。。。for update 添加。

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

行級鎖,每次操作鎖住對應(yīng)的行數(shù)據(jù),鎖定粒度最小,發(fā)生鎖沖突的概率最高,并發(fā)讀最高,應(yīng)用在innodb存儲引擎中。

innodb的數(shù)據(jù)是基于索引組織的,行鎖是通過對索引上的索引項加鎖來實現(xiàn)的,而不是對記錄加的鎖,對于行級鎖,主要分為以下三類:

1、行鎖或者叫record lock記錄鎖,鎖定單個行記錄的鎖,防止其他事物對次行進(jìn)行update和delete操作,在RC,RR隔離級別下都支持。

2、間隙鎖Gap lock,鎖定索引記錄間隙(不含該記錄),確保索引記錄間隙不變,防止其他事物在這個間隙進(jìn)行insert操作,產(chǎn)生幻讀,在RR隔離級別下都支持。

3、臨鍵鎖Next-key-lock,行鎖和間隙鎖組合,同時鎖住數(shù)據(jù),并鎖住數(shù)據(jù)前面的間隙Gap,在RR隔離級別下支持。

innodb實現(xiàn)了以下兩種類型的行鎖

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

2、排他鎖 X: 允許獲取排他鎖的事務(wù)更新數(shù)據(jù),阻止其他事務(wù)獲得相同數(shù)據(jù)集的共享鎖和排他鎖。

insert 語句 排他鎖 自動添加的

update語句 排他鎖 自動添加

delete 語句 排他鎖 自動添加

select 正常查詢語句 不加鎖 。。。

select 。。。lock in share mode 共享鎖 需要手動在select 之后加lock in share mode

select 。。。for update 排他鎖 需要手動在select之后添加for update

默認(rèn)情況下,innodb在repeatable read事務(wù)隔離級別運行,innodb使用next-key鎖進(jìn)行搜索和索引掃描,以防止幻讀。

間隙鎖唯一目的是防止其它事務(wù)插入間隙,間隙鎖可以共存,一個事務(wù)采用的間隙鎖不會阻止另一個事務(wù)在同一間隙上采用的間隙鎖。

MySQL鎖表和解鎖操作

1、確定mysql有鎖表的情況則使用以下命令查看鎖表進(jìn)程

2、殺掉查詢結(jié)果中已經(jīng)鎖表的trx_mysql_thread_id

擴展:

1、查看鎖的事務(wù)

2、查看等待鎖的事務(wù)

3、查詢是否鎖表:

4、查詢進(jìn)程

java程序中如何實現(xiàn)對mysql數(shù)據(jù)庫中表的鎖定

方法1:用mysql命令鎖住表.

public?void?test()?{??

String?sql?=?"lock?tables?aa1?write";??

//?或String?sql?=?"lock?tables?aa1?read";???

//?如果想鎖多個表?lock?tables?aa1?read?,aa2?write?,?.....???

String?sql1?=?"select?*?from?aa1?";??

String?sql2?=?"unlock?tables";??

try?{??

this.pstmt?=?conn.prepareStatement(sql);??

this.pstmt1?=?conn.prepareStatement(sql1);??

this.pstmt2?=?conn.prepareStatement(sql2);??

pstmt.executeQuery();??

pstmt1.executeQuery();??

pstmt2.executeQuery();??

}?catch?(Exception?e)?{??

System.out.println("異常"?+?e.getMessage());??

}??

}

對于read lock 和 write lock官方說明:

1.如果一個線程獲得一個表的READ鎖定,該線程(和所有其它線程)只能從該表中讀取。

如果一個線程獲得一個表的WRITE鎖定,只有保持鎖定的線程可以對表進(jìn)行寫入。

其它的線程被阻止,直到鎖定被釋放時為止。

2.當(dāng)您使用LOCK TABLES時,您必須鎖定您打算在查詢中使用的所有的表。

雖然使用LOCKTABLES語句獲得的鎖定仍然有效,但是您不能訪問沒有被此語句鎖定的任何的表。

同時,您不能在一次查詢中多次使用一個已鎖定的表——使用別名代替,

在此情況下,您必須分別獲得對每個別名的鎖定。

對與read lock 和 write lock個人說明:

1.read lock 和 write lock 是線程級(表級別).

2.在同一個會話中加了read lock鎖. 只能對這個表進(jìn)行讀操作.對這個表以外的任何表都無法進(jìn)行增、刪、改、查的操作.

但是在不同會話中,只能對加了read lock的表進(jìn)行讀操作.但可以對read lock以外的表進(jìn)行增、刪、改、查的操作.

3.在同一個會話中加了write lock鎖.只能對這個表進(jìn)行讀、寫操作.對這個表以外的任何表都無法進(jìn)行增、刪、改、查的操作.

但是在不同會話中,無法對加了write lock的表進(jìn)行讀、寫操作.但可以對write lock以外的表進(jìn)行增、刪、改、查的操作.

4.如果表中使用了別名.(SELECT * FROM aa1 AS byname_table)

在對aa1加鎖時,必須把別名加上去(lock tables aa1 as byname_table read)

在同一個會話中.必須使用別名進(jìn)行查詢.

在不同的會話中.可以不需要使用別名進(jìn)行查詢.

5.在多個會話中可以對同一個表進(jìn)行l(wèi)ock read操作.但不能在多個會話中對同一個表進(jìn)行l(wèi)ock write操作(這些鎖將等待已鎖的表釋放自身的線程鎖)

如果多個會話對同一個表進(jìn)行l(wèi)ock read操作.那么在這些會話中,也只能對以鎖的表進(jìn)行讀操作.

6.如果要你鎖住了一個表,需要嵌套查詢.你必須使用別名,并且,要鎖定別名.

例如.lock table aa1 read ,aa1 as byname_table read;

select * from aa1 where id in (select * from aa1 as xx??where id=2);

7.解鎖必須用unlock tables;

另:

在JAVA程序中,要想解鎖,需要調(diào)用 unlock tables來解鎖.

如果沒有調(diào)用unlock tables.

關(guān)閉connection 、程序結(jié)束 、調(diào)用GC 都能解鎖.

方法2:用記錄鎖鎖表.

public?void?test()?{??

String?sql?=?"select?*?from?aa1?for?update";???

//?select?*?from?aa1?lock?in?share?mode;???

try?{??

conn.setAutoCommit(false);??

this.pstmt?=?conn.prepareStatement(sql);??

pstmt.executeQuery();??

}?catch?(Exception?e)?{??

System.out.println("異常"?+?e.getMessage());??

}??

}

1.for update 與 lock in share mode 屬于行級鎖和頁級鎖

2.for update 排它鎖,lock in share mode 共享鎖

3.對于記錄鎖.必須開啟事務(wù).

4.行級鎖定事實上是索引記錄的鎖定.只要是用索引掃描的行(或沒索引全表掃描的行),都將被鎖住.

5.在不同的隔離級別下還會使用next-key locking算法.即所掃描的行之間的“間隙”也會也鎖住(在Repeatable read和Serializable隔離級別下有間隙鎖).

6.在mysql中共享鎖的含義是:在被共享鎖鎖住的行,即使內(nèi)容被修改且并沒有提交.在另一個會話中依然看到最新修改的信息.

在同一會話中加上了共享鎖.可以對這個表以及這個表以外的所有表進(jìn)行增、刪、改、查的操作.

在不同的會話中.可以查到共享鎖鎖住行的最新消息.但是在Read Uncommitted隔離級別下不能對鎖住的表進(jìn)行刪,

改操作.(需要等待鎖釋放才能操作...)

在Read Committed隔離級別下不能對鎖住的表進(jìn)行刪,改操作.(需要等待鎖釋放才能操作...)

在Repeatable read隔離級別下不能對鎖住行進(jìn)行增、刪、改操作.(需要等待鎖釋放才能操作...)

在Serializable隔離級別下不能對鎖住行進(jìn)行增、刪、改操作.??(需要等待鎖釋放才能操作...)

7.在mysql中排他鎖的含義是:在被排它鎖鎖住的行,內(nèi)容修改并沒提交,在另一個會話中不會看到最新修改的信息。

在不同的會話中.可以查到共享鎖鎖住行的最新消息.但是Read Uncommitted隔離級別下不能對鎖住的表進(jìn)行刪,

改操作.(需要等待鎖釋放才能操作...)

在Read Committed隔離級別下不能對鎖住的表進(jìn)行刪,改操作.(需要等待鎖釋放才能操作...)

在Repeatable read隔離級別下不能對鎖住行進(jìn)行增、刪、改操作.(需要等待鎖釋放才能操作...)

在Serializable隔離級別下不能對鎖住行進(jìn)行增、刪、改操作. (需要等待鎖釋放才能操作...)

8.在同一個會話中的可以疊加多個共享鎖和排他鎖.在多個會話中,需要等待鎖的釋放.

9.SQL中的update 與 for update是一樣的原理.

10.等待超時的參數(shù)設(shè)置:innodb_lock_wait_timeout=50 (單位秒).

11.任何可以觸發(fā)事務(wù)提交的命令,都可以關(guān)閉共享鎖和排它鎖.

MySQL數(shù)據(jù)庫如何鎖定和解鎖數(shù)據(jù)庫表

第一步,創(chuàng)建數(shù)據(jù)庫表writer和查看表結(jié)構(gòu),利用SQL語句:

create table writer(

wid int(10),

wno int(10),

wname varchar(20),

wsex varchar(2),

wage int(2)

第二步,向數(shù)據(jù)庫表writer插入五條數(shù)據(jù),插入后查看表里數(shù)據(jù)

第三步,利用鎖定語句鎖定數(shù)據(jù)庫表writer,利用SQL語句:

lock table writer read;

讓數(shù)據(jù)庫表只讀不能進(jìn)行寫

第四步,為了驗證鎖定效果,可以查看數(shù)據(jù)庫表數(shù)據(jù),利用SQL語句:

select * from writer;

第五步,利用update語句對id=5進(jìn)行更新,SQL語句為:

update writer set wname = '胡思思' where id = 5;

第六步,利用unlock進(jìn)行解鎖,SQL語句為:

unlock tables;

網(wǎng)站名稱:mysql表鎖怎么鎖 mysql鎖表語句
網(wǎng)站路徑:http://muchs.cn/article40/dohecho.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供ChatGPT微信公眾號、網(wǎng)站營銷、小程序開發(fā)、搜索引擎優(yōu)化、網(wǎng)頁設(shè)計公司

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

h5響應(yīng)式網(wǎng)站建設(shè)