在 MySQL 中,一個事務是一組原子性操作的集合,要么全部執行成功,要么全部失敗回滾。但當兩個事務同時執行時,就可能會發生一些不可預知的結果,特別是涉及到并發讀寫的情況。
下面是一個例子,展示了兩個事務同時執行可能導致的問題:
-- 事務 1 START TRANSACTION; UPDATE products SET stock = stock - 1 WHERE id = 1; -- 稍等片刻 SELECT stock FROM products WHERE id = 1; COMMIT; -- 事務 2 START TRANSACTION; UPDATE products SET stock = stock - 1 WHERE id = 1; COMMIT;
在這個例子中,兩個事務都試圖從 id 為 1 的產品庫存中減去 1。如果這兩個事務同時執行,可能會導致庫存減少 2,而不是預期的 1。
為了避免這種情況,可以使用 MySQL 提供的鎖機制來保證事務的原子性。
-- 事務 1 START TRANSACTION; SELECT stock FROM products WHERE id = 1 FOR UPDATE; UPDATE products SET stock = stock - 1 WHERE id = 1; COMMIT; -- 事務 2 START TRANSACTION; SELECT stock FROM products WHERE id = 1 FOR UPDATE; UPDATE products SET stock = stock - 1 WHERE id = 1; COMMIT;
在這個例子中,使用了SELECT ... FOR UPDATE
語句來獲取對 id 為 1 的產品庫存的排他鎖,保證了事務 1 和事務 2 之間的互斥性。如果一個事務正在執行SELECT ... FOR UPDATE
語句,那么其他事務就必須等待,直到該事務釋放鎖為止。
總的來說,在 MySQL 中,保證事務的原子性是非常重要的。使用鎖機制可以避免并發執行時發生不可預知的結果,從而保證系統的穩定性和數據的完整性。