MySQL 中的間隙鎖是為了防止高并發(fā)情況下出現(xiàn)的幽靈讀問(wèn)題,即一個(gè)事務(wù)先查詢(xún)到值,但是提交時(shí)事務(wù)卻已經(jīng)被其他事務(wù)修改了,導(dǎo)致出現(xiàn)數(shù)據(jù)不一致的情況。下面我們來(lái)看看在 MySQL 中,什么情況會(huì)加間隙鎖。
SELECT * FROM `table_name` WHERE `id` >5 FOR UPDATE;
以上 SQL 查詢(xún)語(yǔ)句查詢(xún)了 `table_name` 表中 `id` 大于 5 的記錄,而加了 FOR UPDATE 鎖定了這些記錄,以避免其他事務(wù)對(duì)它們進(jìn)行修改。但是,在這個(gè)查詢(xún)語(yǔ)句中,MySQL 會(huì)在 `id` 為 5 和 6 之間加入一個(gè)間隙鎖,以防止其他事務(wù)在這個(gè)范圍內(nèi)進(jìn)行插入操作。如果想要消除這個(gè)間隙鎖,則需要將查詢(xún)條件改為 `id >= 5`。
SELECT * FROM `table_name` WHERE `id` BETWEEN 3 AND 8;
以上 SQL 查詢(xún)語(yǔ)句查詢(xún)了 `table_name` 表中 `id` 在 3 到 8 之間的記錄,但是 MySQL 會(huì)加入間隙鎖以防止其他事務(wù)在這個(gè)范圍內(nèi)進(jìn)行插入操作。如果想要消除這個(gè)間隙鎖,則需要將查詢(xún)條件改為 `id >= 3 AND id<= 8`。
INSERT INTO `table_name` (`id`, `name`, `age`) VALUES (7, 'Tom', 22);
以上 SQL 語(yǔ)句進(jìn)行了一條插入操作,但是如果在表中已經(jīng)存在 id 為 7 的記錄,則 MySQL 會(huì)對(duì)其進(jìn)行加鎖以避免幽靈讀的問(wèn)題。如果想要消除這個(gè)間隙鎖,則需要在插入之前先進(jìn)行一次查詢(xún)操作,判斷是否已經(jīng)存在相同的記錄。