搜尋
首頁後端開發PHP問題PHP中如何使用PDO操作事務

本篇文章要為大家介紹PHP中使用PDO操作事務的方法。有一定的參考價值,有需要的朋友可以參考一下,希望對大家有幫助。

PHP中如何使用PDO操作事務

關於交易的問題,我們就不多解釋了,以後在學習 MySQL 的相關內容時再深入的了解。今天我們主要是對 PDO 中操作事務的一些小測試,或許能發現一些比較好玩的內容。

在 MyISAM 上使用交易會怎麼樣?

首先,相信只要是學過一點點的 MySQL 相關知識的人都知道,在 MySQL 中常用的兩種表格類型就是 InnoDB 和 MyISAM 這兩種類型。當然,我們今天也不講它們全部的區別,但有一個區別是最明顯的,那就是 MyISAM 不支援事務。那麼,如果我們在 PDO 操作中對 MyISAM 進行事務操作會怎麼樣呢?

// myisam
try {
    $pdo->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
    $pdo->beginTransaction();
    $pdo->exec("insert into tran_myisam (name, age) values ('Joe', 12)");
    $pdo->exec("insert into tran_myisam2 (name, age) values ('Joe', 12, 33)");

    // sleep(30);
    $pdo->commit();

} catch (Exception $e) {
    $pdo->rollBack();
    echo "Failed: " . $e->getMessage(), PHP_EOL;
}

tran_myisam 和 tran_myisam2 表都是 MyISAM 類型的表,在這段程式碼中,我們故意寫錯了 tran_myisam2 的插入語句,讓它走到 catch 中。實際執行的結果是,報錯資訊正常輸出,tran_myisam 表的資料也被插入了。也就是說,針對 MyISAM 表的事務操作是沒有效果的。當然,PDO 也不會主動報錯,如果我們讓第二個 SQL 語句也是正常語句的話,PDO 只會正常執行結束,不會有任何的錯誤或提示訊息。

// innodb
try {
    $pdo->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
    $pdo->beginTransaction();
    $pdo->exec("insert into tran_innodb (name, age) values ('Joe', 12)");
    $pdo->exec("insert into tran_innodb2 (name, age) values ('Joe', 12, 3)");
    // sleep(30);
    $pdo->commit();

} catch (Exception $e) {
    $pdo->rollBack();
    echo "Failed: " . $e->getMessage(), PHP_EOL;
}

我們可以開啟 sleep(30); 這行程式碼的註釋,也就是在交易提交前暫停 30 秒,然後在 MySQL 中查看 infomation_schema.INNODB_TRX 表。這個表中顯示的就是正在執行中的事務。在 InnoDB 類型的資料表執行時就可以看到一則交易正在執行的記錄,而 MyISAM 類型的表中則不會看到任何資訊。

不提交不回滾交易會發生什麼事?

假設我們忘了 commit() ,同時也沒有報錯,這條語句會執行成功嗎?就像下面這段程式碼一樣。

try {
    $pdo->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
    $pdo->beginTransaction();
    $pdo->exec("insert into tran_innodb (name, age) values ('Joe', 12)");
    $pdo->exec("insert into tran_innodb2 (name, age) values ('Joe', 12)");

    // 忘记写 $pdo->commit(); 了
} catch (Exception $e) {
    $pdo->rollBack();
    echo "Failed: " . $e->getMessage(), PHP_EOL;
}

PHP 會在腳本執行結束後,其實也就是在 $pdo 物件析構時回滾這個交易。也就是說,這裡的 SQL 語句是不會執行的。但是,盡量不要這麼做,因為在正式環境中,我們的程式碼非常複雜,而且不一定會析構成功。這樣的話,可能會有長時間佔據的事務存在,最終結果就是會導致 MySQL 的 IPQS 奇高,而且還很難找到原因。所以,在使用事務的時候,一定要記得 commit() 和 rollBack() 都是我們的親兄弟,絕對不能落下他們。

上一個交易沒有提交沒有回滾,下一個交易會執行嗎?

同樣的,在上一個問題的基礎上我們再繼續延伸。如果有兩個事務依序執行,第一個事務沒有提交,沒有回滾,那麼下一個事務還能執行嗎?

// innodb
try {
    $pdo->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
    $pdo->beginTransaction();
    $pdo->exec("insert into tran_innodb (name, age) values ('Joe', 12)");
    $pdo->exec("insert into tran_innodb2 (name, age) values ('Joe', 12)");
    // 忘记写 $pdo->commit(); 了
} catch (Exception $e) {
    $pdo->rollBack();
    echo "Failed: " . $e->getMessage(), PHP_EOL;
}

// innodb
try {
    $pdo->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
    $pdo->beginTransaction();
    $pdo->exec("insert into tran_innodb (name, age) values ('BW', 12)");
    $pdo->exec("insert into tran_innodb2 (name, age) values ('BW', 12)");

    // sleep(30);
    $pdo->commit();
} catch (Exception $e) {
    $pdo->rollBack();
    echo "Failed: " . $e->getMessage(), PHP_EOL; // Failed: There is already an active transaction
}

我們可以看到,第二段事務直接就報錯了,內容是:「這裡有一個已經存在的活動事務」。也就是說如果上一個交易沒有提交沒有回滾的話,第二個交易是無法執行的。

總結

今天我們只是學習並測試了幾個事務相關的小問題,但問題雖小卻有可能帶來嚴重的線上事故,大家在開發的時候一定要小心。關於事務的詳細內容在未來深入學習 MySQL 的時候我們再好好研究。

測試程式碼:

https://github.com/zhangyue0503/dev-blog/blob/master/php/202008/source/PHP中使用PDO操作事务的一些小测试.php

推薦學習:php影片教學

#

以上是PHP中如何使用PDO操作事務的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述
本文轉載於:segmentfault。如有侵權,請聯絡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

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

熱工具

MantisBT

MantisBT

Mantis是一個易於部署的基於Web的缺陷追蹤工具,用於幫助產品缺陷追蹤。它需要PHP、MySQL和一個Web伺服器。請查看我們的演示和託管服務。

EditPlus 中文破解版

EditPlus 中文破解版

體積小,語法高亮,不支援程式碼提示功能

SublimeText3 英文版

SublimeText3 英文版

推薦:為Win版本,支援程式碼提示!

SublimeText3 Linux新版

SublimeText3 Linux新版

SublimeText3 Linux最新版

記事本++7.3.1

記事本++7.3.1

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