検索
ホームページバックエンド開発PHPチュートリアルPHP 同時実行シナリオの 3 つのソリューション

PHP 同時実行シナリオの 3 つのソリューション

フラッシュ セールやラッシュ セールなどの同時実行シナリオでは、過剰販売が発生する可能性があります。PHP 言語にはネイティブの同時実行ソリューションがないため、それを実現するには他の方法が必要です。同時実行制御。

リストされている一般的な解決策は次のとおりです:

キューを使用し、キューを処理する追加のプロセスを設定し、すべての同時リクエストをキューに入れ、追加のプロセスによってそれらをシリアルに処理します。同時実行の問題は存在しません。ただし、追加のプロセス サポートと重大な処理遅延が必要になります。この記事では、この方法については最初に説明しません。

データベースのトランザクション特性を使用してアトミック更新を実行します。この方法は、データベースのトランザクション特性に依存する必要があります。

ファイル排他ロックを使用すると、注文リクエストを処理するときに flock を使用してファイルをロックできます。ロックを正常に取得した人だけが注文を処理できます。

1. Redis トランザクション機能の活用

Redis トランザクションはアトミックな操作であり、注文処理中に他の同時プロセスによってデータが変更されないようにすることができます。

サンプルコード:

<?php
$http = new swoole_http_server("0.0.0.0", 9509);   // 监听 9509
$http->set(array(
    &#39;reactor_num&#39; => 2,  //reactor thread num
    &#39;worker_num&#39; => 4    //worker process num
));
$http->on(&#39;request&#39;, function (swoole_http_request $request, swoole_http_response $response) {
    $uniqid = uniqid(&#39;uid-&#39;, TRUE);    // 模拟唯一用户ID
    $redis = new Redis();
    $redis->connect(&#39;127.0.0.1&#39;, 6379);    // 连接 redis
    $redis->watch(&#39;rest_count&#39;);  // 监测 rest_count 是否被其它的进程更改
    $rest_count = intval($redis->get("rest_count"));  // 模拟唯一订单ID
    if ($rest_count > 0){
        $value = "{$rest_count}-{$uniqid}";  // 表示当前订单,被当前用户抢到了
        // do something ... 主要是模拟用户抢到单后可能要进行的一些密集运算
        $rand  = rand(100, 1000000);
        $sum = 0;
        for ($i = 0; $i < $rand; $i++) {$sum += $i;}
      // redis 事务
        $redis->multi();
        $redis->lPush(&#39;uniqids&#39;, $value);
        $redis->decr(&#39;rest_count&#39;);
        $replies = $redis->exec();  // 执行以上 redis 事务
      // 如果 rest_count 的值被其它的并发进程更改了,以上事务将回滚
        if (!$replies) {
            echo "订单 {$value} 回滚" . PHP_EOL;
        }
    }
    $redis->unwatch();
});
$http->start();

ab testを使用する

$ ab -t 20 -c 10 http://192.168.1.104:9509/

2.ファイル排他ロックを使用する(ブロッキングモード)

ブロッキングモード他のプロセスがロックを占有しているときにプロセスがファイル排他ロックを取得すると、プロセスはハングして他のプロセスがロックを解放するのを待ち、その後続行する前に自らロックを取得します。

サンプルコード:

<?php
$http = new swoole_http_server("0.0.0.0", 9510);
$http->set(array(
    &#39;reactor_num&#39; => 2,  //reactor thread num
    &#39;worker_num&#39; => 4    //worker process num
));
$http->on(&#39;request&#39;, function (swoole_http_request $request, swoole_http_response $response) {
    $uniqid = uniqid(&#39;uid-&#39;, TRUE);
    $redis = new Redis();
    $redis->connect(&#39;127.0.0.1&#39;, 6379);
    $fp = fopen("lock.txt", "w+");
    // 阻塞(等待)模式, 要取得独占锁定(写入的程序)
    if (flock($fp,LOCK_EX)) {  //锁定当前指针
      // 成功取得锁后,放心处理订单
        $rest_count = intval($redis->get("rest_count"));
        $value = "{$rest_count}-{$uniqid}";
        if ($rest_count > 0) {
            // do something ...
            $rand = rand(100, 1000000);
            $sum = 0;
            for ($i = 0; $i < $rand; $i++) {$sum += $i;}
            $redis->lPush(&#39;uniqids&#39;, $value);
            $redis->decr(&#39;rest_count&#39;);
        }
      // 订单处理完成后,再释放锁
        flock($fp, LOCK_UN);
    }
    fclose($fp);
});
$http->start();

ab testを使用

$ ab -t 20 -c 10 http://192.168.1.104:9510/

3.ファイル排他ロック(ノンブロッキングモード)を使用

非ブロッキング モードでは、他のプロセスがロックを占有しているときにプロセスがファイルの排他ロックを取得した場合、プロセスはロックの取得が失敗したと即座に判断し、実行を続行します。 \

サンプル コード:

<?php
$http = new swoole_http_server("0.0.0.0", 9511);
$http->set(array(
    &#39;reactor_num&#39; => 2,  //reactor thread num
    &#39;worker_num&#39; => 4    //worker process num
));
$http->on(&#39;request&#39;, function (swoole_http_request $request, swoole_http_response $response) {
    $uniqid = uniqid(&#39;uid-&#39;, TRUE);
    $redis = new Redis();
    $redis->connect(&#39;127.0.0.1&#39;, 6379);
    $fp = fopen("lock.txt", "w+");
    // 非阻塞模式, 如果不希望 flock() 在锁定时堵塞,则给 lock 加上 LOCK_NB
    if(flock($fp,LOCK_EX | LOCK_NB))   //锁定当前指针
    {
      // 成功取得锁后,放心处理订单
        $rest_count = intval($redis->get("rest_count"));
        $value = "{$rest_count}-{$uniqid}";
        if($rest_count > 0){
            // do something ...
            $rand  = rand(100, 1000000);
            $sum=0;
            for ($i=0;$i<$rand;$i++){ $sum+=$i; }
            $redis->lPush(&#39;uniqids&#39;, $value);
            $redis->decr(&#39;rest_count&#39;);
        }
      // 订单处理完成后,再释放锁
        flock($fp,LOCK_UN);
    } else {
      // 如果获取锁失败,马上进入这里执行
        echo "{$uniqid} - 系统繁忙,请稍后再试".PHP_EOL;
    }
    fclose($fp);
});
$http->start();

Use ab test

$ ab -t 20 -c 10 http://192.168.1.104:9511/

最後に、3 つの処理メソッドのテスト結果の比較を示します。

redis トランザクション メソッド:

......
Concurrency Level:      10
Time taken for tests:   20.005 seconds
Complete requests:      17537
Failed requests:        0
Total transferred:      2578380 bytes
HTML transferred:       0 bytes
Requests per second:    876.62 [#/sec] (mean)
Time per request:       11.407 [ms] (mean)
Time per request:       1.141 [ms] (mean, across all concurrent requests)
Transfer rate:          125.86 [Kbytes/sec] received
......

ファイル排他ロック (ブロッキングモード):

......
Concurrency Level:      10
Time taken for tests:   20.003 seconds
Complete requests:      8205
Failed requests:        0
Total transferred:      1206282 bytes
HTML transferred:       0 bytes
Requests per second:    410.19 [#/sec] (mean)
Time per request:       24.379 [ms] (mean)
Time per request:       2.438 [ms] (mean, across all concurrent requests)
Transfer rate:          58.89 [Kbytes/sec] received
......

ファイル排他ロック (ノンブロッキングモード):

......
Concurrency Level:      10
Time taken for tests:   20.002 seconds
Complete requests:      8616
Failed requests:        0
Total transferred:      1266846 bytes
HTML transferred:       0 bytes
Requests per second:    430.77 [#/sec] (mean)
Time per request:       23.214 [ms] (mean)
Time per request:       2.321 [ms] (mean, across all concurrent requests)
Transfer rate:          61.85 [Kbytes/sec] received
......

テスト結果を比較した後、redis トランザクション メソッドは次のようになります。ファイル排他ロック方式よりも優れており、ファイル排他ロックモードでは、ノンブロッキングモードの方がブロッキングモードより優れています。

推奨チュートリアル: 「PHP チュートリアル

以上がPHP 同時実行シナリオの 3 つのソリューションの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明
この記事はlearnkuで複製されています。侵害がある場合は、admin@php.cn までご連絡ください。
PHP対Python:違いを理解しますPHP対Python:違いを理解しますApr 11, 2025 am 12:15 AM

PHP and Python each have their own advantages, and the choice should be based on project requirements. 1.PHPは、シンプルな構文と高い実行効率を備えたWeb開発に適しています。 2。Pythonは、簡潔な構文とリッチライブラリを備えたデータサイエンスと機械学習に適しています。

PHP:それは死にかけていますか、それとも単に適応していますか?PHP:それは死にかけていますか、それとも単に適応していますか?Apr 11, 2025 am 12:13 AM

PHPは死にかけていませんが、常に適応して進化しています。 1)PHPは、1994年以来、新しいテクノロジーの傾向に適応するために複数のバージョンの反復を受けています。 2)現在、電子商取引、コンテンツ管理システム、その他の分野で広く使用されています。 3)PHP8は、パフォーマンスと近代化を改善するために、JITコンパイラおよびその他の機能を導入します。 4)Opcacheを使用してPSR-12標準に従って、パフォーマンスとコードの品質を最適化します。

PHPの未来:適応と革新PHPの未来:適応と革新Apr 11, 2025 am 12:01 AM

PHPの将来は、新しいテクノロジーの傾向に適応し、革新的な機能を導入することで達成されます。1)クラウドコンピューティング、コンテナ化、マイクロサービスアーキテクチャに適応し、DockerとKubernetesをサポートします。 2)パフォーマンスとデータ処理の効率を改善するために、JITコンパイラと列挙タイプを導入します。 3)パフォーマンスを継続的に最適化し、ベストプラクティスを促進します。

PHPの抽象クラスまたはインターフェイスに対して、いつ特性を使用しますか?PHPの抽象クラスまたはインターフェイスに対して、いつ特性を使用しますか?Apr 10, 2025 am 09:39 AM

PHPでは、特性は方法が必要な状況に適していますが、継承には適していません。 1)特性により、クラスの多重化方法が複数の継承の複雑さを回避できます。 2)特性を使用する場合、メソッドの競合に注意を払う必要があります。メソッドの競合は、代替およびキーワードとして解決できます。 3)パフォーマンスを最適化し、コードメンテナビリティを改善するために、特性の過剰使用を避け、その単一の責任を維持する必要があります。

依存関係噴射コンテナ(DIC)とは何ですか?また、なぜPHPで使用するのですか?依存関係噴射コンテナ(DIC)とは何ですか?また、なぜPHPで使用するのですか?Apr 10, 2025 am 09:38 AM

依存関係噴射コンテナ(DIC)は、PHPプロジェクトで使用するオブジェクト依存関係を管理および提供するツールです。 DICの主な利点には、次のものが含まれます。1。デカップリング、コンポーネントの独立したもの、およびコードの保守とテストが簡単です。 2。柔軟性、依存関係を交換または変更しやすい。 3.テスト可能性、単体テストのために模擬オブジェクトを注入するのに便利です。

通常のPHPアレイと比較して、SPL SPLFIXEDARRAYとそのパフォーマンス特性を説明してください。通常のPHPアレイと比較して、SPL SPLFIXEDARRAYとそのパフォーマンス特性を説明してください。Apr 10, 2025 am 09:37 AM

SplfixedArrayは、PHPの固定サイズの配列であり、高性能と低いメモリの使用が必要なシナリオに適しています。 1)動的調整によって引き起こされるオーバーヘッドを回避するために、作成時にサイズを指定する必要があります。 2)C言語アレイに基づいて、メモリと高速アクセス速度を直接動作させます。 3)大規模なデータ処理とメモリに敏感な環境に適していますが、サイズが固定されているため、注意して使用する必要があります。

PHPは、ファイルを安全に処理する方法をどのように処理しますか?PHPは、ファイルを安全に処理する方法をどのように処理しますか?Apr 10, 2025 am 09:37 AM

PHPは、$ \ _ファイル変数を介してファイルのアップロードを処理します。セキュリティを確保するための方法には次のものが含まれます。1。アップロードエラー、2。ファイルの種類とサイズを確認する、3。ファイル上書きを防ぐ、4。ファイルを永続的なストレージの場所に移動します。

Null Coulescingオペレーター(??)およびNull Coulescing Assignment Operator(?? =)とは何ですか?Null Coulescingオペレーター(??)およびNull Coulescing Assignment Operator(?? =)とは何ですか?Apr 10, 2025 am 09:33 AM

JavaScriptでは、nullcoalescingoperator(??)およびnullcoalescingsignmentoperator(?? =)を使用できます。 1.??最初の非潜水金または非未定されたオペランドを返します。 2.??これらの演算子は、コードロジックを簡素化し、読みやすさとパフォーマンスを向上させます。

See all articles

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

AI Hentai Generator

AI Hentai Generator

AIヘンタイを無料で生成します。

ホットツール

Dreamweaver Mac版

Dreamweaver Mac版

ビジュアル Web 開発ツール

EditPlus 中国語クラック版

EditPlus 中国語クラック版

サイズが小さく、構文の強調表示、コード プロンプト機能はサポートされていません

WebStorm Mac版

WebStorm Mac版

便利なJavaScript開発ツール

SAP NetWeaver Server Adapter for Eclipse

SAP NetWeaver Server Adapter for Eclipse

Eclipse を SAP NetWeaver アプリケーション サーバーと統合します。

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)