Locker 協程鎖

Locker是一種內存鎖,用於協程間的同步,常用來在協程中排隊訪問某種臨界資源,例如某個數據庫組件沒有做連接池,則可以通過Locker來排隊使用該組件,避免因為多個協程同時使用同一個連接資源導致數據異常。

提示
此特性需要 workerman>=5.1.0

注意

  • Locker支持Swoole/Swow/Fiber/Select/Event驅動
  • Locker是用於同一個進程的不同協程間排隊互斥訪問某個資源的,進程與進程間互不影響
<?php
use Workerman\Connection\TcpConnection;
use Workerman\Coroutine\Locker;
use Workerman\Events\Swoole;
use Workerman\Protocols\Http\Request;
use Workerman\Worker;
require_once __DIR__ . '/vendor/autoload.php';

$worker = new Worker('http://0.0.0.0:8001');

$worker->eventLoop = Swoole::class; // Or Swow::class or Fiber::class

$worker->onMessage = function (TcpConnection $connection, Request $request) {
    static $redis;
    if (!$redis) {
        $redis = new Redis();
        $redis->connect('127.0.0.1', 6379);
    }
    // 避免多個協程同時使用同一個連接,發生類似 "Socket#10 has already been bound to another coroutine" 錯誤
    Locker::lock('redis');
    $time = $redis->time();
    Locker::unlock('redis');
    $connection->send(json_encode($time));
};

Worker::runAll();

介面說明

interface LockerInterface
{
    /**
     * 加鎖
     */
    public static function lock(string $key): bool

    /**
     * 解鎖
     */
    public static function unlock(string $key): bool
}