Swoole 中使用 Table 內存表實現進程間共享數據


背景

在多進程模式下進程之間的內存是相互隔離的,在一個工作進程中的全局變量和超全局變量,在另一個工作進程中是無法讀取和操作的。

如果只有一個工作進程,則不存在進程隔離問題,可以使用全局變量和超全局變量。

要實現進程間共享數據,我們可以使用第三方的 Redis 內存數據庫或 Swoole 內置的 Table 共享內存來實現。

Table 的優勢

  • 性能強悍,單線程每秒可讀寫 200 萬次;
  • 應用代碼無需加鎖,Table 內置行鎖自旋鎖,所有操作均是多線程 / 多進程安全。用戶層完全不需要考慮數據同步問題;
  • 支持多進程,Table 可以用於多進程之間共享數據;
  • 使用行鎖,而不是全局鎖,僅當 2 個進程在同一 CPU 時間,並發讀取同一條數據才會進行發生搶鎖。

使用示例:

<?php
// 實例化一個占用的共享內存大小為1024的內存表
$table = new Swoole\Table(1024);
// 內存表增加3列
$table->column('fd', Swoole\Table::TYPE_INT);
$table->column('reactor_id', Swoole\Table::TYPE_INT);
$table->column('data', Swoole\Table::TYPE_STRING, 64);
$table->create();

$serv = new Swoole\Server('127.0.0.1', 9501);
// 設置數據包分發策略:輪循模式
$serv->set(['dispatch_mode' => 1]);
$serv->table = $table;

$serv->on('receive', function ($serv, $fd, $reactor_id, $data) {
    $cmd = explode(" ", trim($data));

    if ($cmd[0] == 'get') {
        //get self
        if (count($cmd) < 2) {
            $cmd[1] = $fd;
        }
        $get_fd = intval($cmd[1]);
        $info = $serv->table->get($get_fd);
        $serv->send($fd, var_export($info, true) . "\n");
    } elseif ($cmd[0] == 'set') {
        // 使用連接的文件描述符作為key寫入內存表
        $ret = $serv->table->set($fd, array('fd' => $fd, 'reactor_id' => $reactor_id, 'data' => $cmd[1]));
        if ($ret === false) {
            $serv->send($fd, "ERROR\n");
        } else {
            $serv->send($fd, "OK\n");
        }
    } else {
        $serv->send($fd, "command error.\n");
    }
});

$serv->start();


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM