Type something to search...
MySQL 你到底是在鎖屁喔

MySQL 你到底是在鎖屁喔

  • Database
  • 03 May, 2020

剛剛在研究這個簡單的範例:

CREATE TABLE `posts` (
 `id` bigint(20) NOT NULL AUTO_INCREMENT,
 `content` varchar(255) DEFAULT NULL,
 `user_id` int(11) DEFAULT NULL,
 PRIMARY KEY (`id`),
 KEY `index_posts_on_user_id` (`user_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
INSERT INTO `posts` (`id`, `content`, `user_id`)
VALUES
 (1,’All Might’,1),
 (2,’All Might’,2),
 (3,’All Might’,3),
 (4,’All Might’,4),
 (5,’All Might’,5);

Session A:

BEGIN;
SELECT `posts`.* FROM `posts` WHERE `posts`.`user_id` IN (1,2) FOR UPDATE;

Session B:

SELECT `posts`.* FROM `posts` WHERE `posts`.`user_id` = 2 FOR UPDATE

當 Session A 還沒結束 transaction 時,Session B 進入了 Lock Wait。posts`.`user_id有打 index,照理來說應該不會鎖住才對,因為走 index_posts_on_user_id 的 index tree 根本不會碰到 user_id = 5 的 index。

追查了之後才發現,因為表太小,MySQL 沒有用 index_posts_on_user_id index,而是走 PRIMARY KEY 的 index tree。所以就掃描整張表,導致所有 record 都被鎖住了。

可以把上面範例中的 Session A ,分別改用這兩種來比較就知道差異了:

BEGIN;
SELECT `posts`.* FROM `posts` FORCE INDEX(PRIMARY) WHERE `posts`.`user_id` IN (1,2) FOR UPDATE;
BEGIN;
SELECT `posts`.* FROM `posts` FORCE INDEX(index_posts_on_user_id) WHERE `posts`.`user_id` IN (1,2) FOR UPDATE;

Related Posts

INSERT…ON DUPLICATE KEY UPDATE ..在 MySQL 5.7.26(Aurora MySQL 2.10.3) 後的行為差異
INSERT…ON DUPLICATE KEY UPDATE ..在 MySQL 5.7.26(Aurora MySQL 2.10.3) 後的行為差異

MySQL 5.7.26 修復了一個 bug #2596684([release note](https://dev.mysql.com/doc/relnotes/mysql/5.7/en/news-5-7-26.html

read more
[MySQL] 使用 update_all & subquery 可能會鎖住全表
[MySQL] 使用 update_all & subquery 可能會鎖住全表

團隊最近遇到一個 subquery + update_all 會鎖整張表的問題,花了一點時間研究 注意:以下為 MySQL 5.6 的實驗探討,在 MySQL 8.0.21 的這個機制已經被優化了 問題 假設有一份 schema 長這樣,有 users 和 posts 兩張表。 create_table "posts", options: "ENGINE=

read more
[MySQL] 對不存在的 row 加鎖,會造成 Deadlock
[MySQL] 對不存在的 row 加鎖,會造成 Deadlock

[MySQL] 對不存在的 row 加鎖,會造成 Deadlock 一個範例 初始資料設定 CREATE TABLE `users` ( `id` int(11) unsigned NOT NULL AUTO_INCREMENT, `name` varchar(100) COLLATE utf8mb4_unicode_ci NOT NULL DEFAUL

read more
[MySQL] 跑個 migration 怎麼就倒站了
[MySQL] 跑個 migration 怎麼就倒站了

如果網站正在 migrate 時,剛好遇到一個 transaction 沒有結束,就算只是 SELECT ,也有可能讓網站倒站。 一個血淋淋的例子 某個沒有經驗的工程師 A: 對 users table 做一些簡單的查詢,但是開啟了 transaction 沒有 commit BEGIN; SELECT * FROM users; ...**正在

read more
Postgresql 常用命令
Postgresql 常用命令

$createdb [database_name]$pg_dump [database_name] -f [backup_filename]$psql -f [backup_filename] [database_name]$dropdb [database_name]

read more
在 MySQL 5.6 使用 GROUP BY 的筆記
在 MySQL 5.6 使用 GROUP BY 的筆記

在 MySQL 5.6,如果使用了 GROUP BY A 欄位,但是 SELECT 的 **B 欄位,沒有加上 aggregate function,且群組裡 row 彼此間 B 欄位 **的值並不相同,則會以 nondeterministic 的方式決定要取出 **B 欄位 **的值。 可參考文件:[https://dev.mysql.com/doc/refman/5.6/en

read more
[Rails][MySQL] 使用 update_all + JOIN + LIMIT 需注意效能問題
[Rails][MySQL] 使用 update_all + JOIN + LIMIT 需注意效能問題

MySQL 使用 JOIN + update_all 時,若遇到 LIMIT 則會轉為 subquery,需注意效能問題。 Rails 在使用 update_all 時,如果有用到 JOIN,會改用 subquery 的形式改寫,但若 Adapter 為 MySQL 時就會維持,因為 MySQL 語法本身支援 JOIN + UPDATE。但若是包含 LIMIT 時,由於 MySQL 也不支援,所

read more