PHP幾種回調寫法
PHP裡通過匿名函數寫回調是最方便的,但除了匿名函數的方式,PHP還有其它的回調寫法。以下是PHP幾種回調寫法的示例。
1、匿名函數回調
<?php
use Workerman\Worker;
use Workerman\Connection\TcpConnection;
use Workerman\Protocols\Http\Request;
require_once __DIR__ . '/vendor/autoload.php';
$http_worker = new Worker("http://0.0.0.0:2345");
// 匿名函數回調
$http_worker->onMessage = function(TcpConnection $connection, Request $data)
{
// 向瀏覽器發送hello world
$connection->send('hello world');
};
Worker::runAll();
2、普通函數回調
<?php
use Workerman\Worker;
use Workerman\Connection\TcpConnection;
use Workerman\Protocols\Http\Request;
require_once __DIR__ . '/vendor/autoload.php';
$http_worker = new Worker("http://0.0.0.0:2345");
// 普通函數回調
$http_worker->onMessage = 'on_message';
// 普通函數
function on_message(TcpConnection $connection, Request $request)
{
// 向瀏覽器發送hello world
$connection->send('hello world');
}
Worker::runAll();
3、類方法作為回調
MyClass.php
use Workerman\Worker;
use Workerman\Connection\TcpConnection;
class MyClass{
public function __construct(){}
public function onWorkerStart(Worker $worker){}
public function onConnect(TcpConnection $connection){}
public function onMessage(TcpConnection $connection, $message) {}
public function onClose(TcpConnection $connection){}
public function onWorkerStop(Worker $worker){}
}
啟動腳本 start.php
<?php
use Workerman\Worker;
require_once __DIR__ . '/vendor/autoload.php';
// 載入MyClass
require_once __DIR__.'/MyClass.php';
$worker = new Worker("websocket://0.0.0.0:2346");
// 創建一個對象
$my_object = new MyClass();
// 調用類的方法
$worker->onWorkerStart = array($my_object, 'onWorkerStart');
$worker->onConnect = array($my_object, 'onConnect');
$worker->onMessage = array($my_object, 'onMessage');
$worker->onClose = array($my_object, 'onClose');
$worker->onWorkerStop = array($my_object, 'onWorkerStop');
Worker::runAll();
注意:
以上的代碼結構不允許在構造函數裡初始化資源(MySQL 連接、Redis 連接、Memcache 連接等),因為$my_object = new MyClass();運行在主進程。以 MySQL 為例,在主進程初始化的 MySQL 連接等資源會被子進程繼承,每個子進程都可以操作這個數據庫連接,但這些連接在 MySQL 服務端對應的是同一個連接,會發生不可預期的錯誤,例如mysql gone away錯誤。
以上代碼結構如果需要在類的構造函數裡初始化資源,可以採用以下寫法。
MyClass.php
use Workerman\Worker;
use Workerman\Connection\TcpConnection;
class MyClass{
protected $db = null;
public function __construct(){
// 假設數據庫連接類是MyDbClass
$db = new MyDbClass();
}
public function onWorkerStart(Worker $worker){}
public function onConnect(TcpConnection $connection){}
public function onMessage(TcpConnection $connection, $message) {}
public function onClose(TcpConnection $connection){}
public function onWorkerStop(Worker $worker){}
}
啟動腳本 start.php
<?php
use Workerman\Worker;
require_once __DIR__ . '/vendor/autoload.php';
$worker = new Worker("websocket://0.0.0.0:2346");
// 在onWorkerStart裡初始化類
$worker->onWorkerStart = function($worker) {
// 載入MyClass
require_once __DIR__.'/MyClass.php';
// 創建一個對象
$my_object = new MyClass();
// 調用類的方法
$worker->onConnect = array($my_object, 'onConnect');
$worker->onMessage = array($my_object, 'onMessage');
$worker->onClose = array($my_object, 'onClose');
$worker->onWorkerStop = array($my_object, 'onWorkerStop');
};
Worker::runAll();
上面代碼結構中onWorkerStart運行時已經是屬於子進程,等於每個子進程各自建立自己的 MySQL 連接,所以不會有共享連接的問題。這樣還有一個好處就是支持業務代碼 reload。因為 MyClass.php 是在子進程載入的,根據 reload 規則,業務更改 MyClass.php 後直接 reload 即可生效。
4、類的靜態方法作為回調
靜態類 MyClass.php
use Workerman\Worker;
use Workerman\Connection\TcpConnection;
class MyClass{
public static function onWorkerStart(Worker $worker){}
public static function onConnect(TcpConnection $connection){}
public static function onMessage(TcpConnection $connection, $message) {}
public static function onClose(TcpConnection $connection){}
public static function onWorkerStop(Worker $worker){}
}
啟動腳本 start.php
<?php
use Workerman\Worker;
require_once __DIR__ . '/vendor/autoload.php';
// 載入MyClass
require_once __DIR__.'/MyClass.php';
$worker = new Worker("websocket://0.0.0.0:2346");
// 調用類的靜態方法。
$worker->onWorkerStart = array('MyClass', 'onWorkerStart');
$worker->onConnect = array('MyClass', 'onConnect');
$worker->onMessage = array('MyClass', 'onMessage');
$worker->onClose = array('MyClass', 'onClose');
$worker->onWorkerStop = array('MyClass', 'onWorkerStop');
// 如果類帶命名空間,則是類似這樣的寫法
// $worker->onWorkerStart = array('your\namesapce\MyClass', 'onWorkerStart');
// $worker->onConnect = array('your\namesapce\MyClass', 'onConnect');
// $worker->onMessage = array('your\namesapce\MyClass', 'onMessage');
// $worker->onClose = array('your\namesapce\MyClass', 'onClose');
// $worker->onWorkerStop = array('your\namesapce\MyClass', 'onWorkerStop');
Worker::runAll();
注意:根據 PHP 的運行機制,如果沒用調用 new 則不會調用構造函數,另外靜態類的方法裡不允許使用$this。