PDO を使用して PHP でトランザクションを操作するための小さなテストをいくつか共有します

藏色散人
リリース: 2023-04-10 09:30:01
転載
2966 人が閲覧しました

トランザクションの問題についてはこれ以上説明しません。関連する問題を学習するときにさらに詳しく説明します。今後の MySQL の内容についてご理解ください。今日は主に PDO でのトランザクション操作に関する小規模なテストを行っており、興味深い内容が見つかるかもしれません。

MyISAM でトランザクションを使用するとどうなりますか?

まず、MySQL について少し学んだ人なら、MySQL で一般的に使用される 2 つのテーブル タイプが InnoDB と MyISAM であることをご存知だと思います。もちろん、今日はそのすべての違いについて説明するつもりはありませんが、最も明白な違いが 1 つあります。それは、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 の間違った挿入ステートメントを記述し、それをキャッチに入れています。実際の実行結果は、エラーメッセージが正常に出力され、tran_myisam テーブルのデータも挿入されます。つまり、MyISAM テーブルに対するトランザクション操作は効果がありません。もちろん、PDO はエラーを積極的に報告しません。2 番目の 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() が兄弟であり、取り残されてはならないことを覚えておく必要があります。

前のトランザクションはコミットまたはロールバックされませんでした。次のトランザクションは実行されますか?

同様に、前の質問に基づいて拡張を続けます。 2 つのトランザクションが連続して実行され、最初のトランザクションがコミットまたはロールバックされなかった場合、次のトランザクションは引き続き実行できますか?

// 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
}
ログイン後にコピー

2 番目のトランザクションがエラーを直接報告していることがわかります。その内容は、「ここに既存のアクティブなトランザクションがあります。」です。つまり、前のトランザクションがコミットまたはロールバックされていない場合、2 番目のトランザクションは実行できません。

まとめ

今日は、トランザクションに関連するいくつかの小さな問題を学び、テストしました。ただし、問題は小さいとはいえ、オンラインで重大な事故を引き起こす可能性があります。慎重に開発する場合は全員が注意する必要があります。トランザクションの詳細については、今後 MySQL を詳しく学習するときに学習します。

测试代码:
https://github.com/zhangyue0503/dev-blog/blob/master/php/202008/source/PHP中使用PDO操作事务的一些小测试.php
ログイン後にコピー

推奨学習: 「PHP ビデオ チュートリアル

以上がPDO を使用して PHP でトランザクションを操作するための小さなテストをいくつか共有しますの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

関連ラベル:
php
ソース:segmentfault.com
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
最新の問題
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート
私たちについて 免責事項 Sitemap
PHP中国語ウェブサイト:福祉オンライン PHP トレーニング,PHP 学習者の迅速な成長を支援します!