# swoole_server中內存管理機制
swoole_server啟動后內存管理的底層原理與普通php-cli程序一致。具體請參考Zend VM內存管理方面的文章。
## 局部變量
在事件回調函數返回后,所有局部對象和變量會全部回收,不需要unset。如果變量是一個資源類型,那么對應的資源也會被PHP底層釋放。
~~~
function test()
{
$a = new Object;
$b = fopen('/data/t.log', 'r+');
$c = new swoole_client(SWOOLE_SYNC);
$d = new swoole_client(SWOOLE_SYNC);
global $e;
$e['client'] = $d;
}
~~~
* $a, $b, $c 都是局部變量,當此函數return時,這3個變量會立即釋放,對應的內存會理解釋放,打開的IO資源文件句柄會立即關閉。
* $d 也是局部變量,但是return前將它保存到了全局變量$e,所以不會釋放。當執行unset($e['client'])時,并且沒有任何其他PHP變量仍然在引用$d變量,那么$d 就會被釋放。
### 全局變量
在PHP中,有3類全局變量。
* 使用global關鍵詞聲明的變量
* 使用static關鍵詞聲明的類靜態變量、函數靜態變量
* PHP的超全局變量,包括$_GET、$_POST、$GLOBALS等
>全局變量和對象,類靜態變量,保存在swoole_server對象上的變量不會被釋放。需要程序員自行處理這些變量和對象的銷毀工作。
>在swoole中如果想在內存中永久保存某些數據資源,可以將資源放到全局變量中或者類的靜態成員中。
~~~
class Test
{
static $array = array();
static $string = '';
}
function onReceive($serv, $fd, $reactorId, $data)
{
Test::$array[] = $fd;
Test::$string .= $data;
}
~~~
* 在事件回調函數中需要特別注意非局部變量的array類型值,某些操作如 TestClass::$array[] = "string" 可能會造成內存泄漏,嚴重時可能發生爆內存,必要時應當注意清理大數組。
* 在事件回調函數中,非局部變量的字符串進行拼接操作是必須小心內存泄漏,如 TestClass::$string .= $data,可能會有內存泄漏,嚴重時可能發生爆內存。
## 解決方法
* 同步阻塞并且請求響應式無狀態的Server程序可以設置max_request,當Worker進程/Task進程結束運行時或達到任務上限后進程自動退出。該進程的所有變量/對象/資源均會被釋放回收。
* 程序內在onClose或設置定時器及時使用unset清理變量,回收資源
## 異步客戶端
Swoole提供的異步客戶端與普通的PHP變量不同,異步客戶端在發起connect時底層會增加一次引用計數,在連接close時會減少引用計數。
>包括swoole_client、swoole_mysql、swoole_redis、swoole_http_client
~~~
function test()
{
$client = new swoole_client(SWOOLE_TCP | SWOOLE_ASYNC);
$client->on("connect", function($cli) {
$cli->send("hello world\n");
});
$client->on("receive", function($cli, $data){
echo "Received: ".$data."\n";
$cli->close();
});
$client->on("error", function($cli){
echo "Connect failed\n";
});
$client->on("close", function($cli){
echo "Connection close\n";
});
$client->connect('127.0.0.1', 9501);
return;
}
~~~
* $client是局部變量,常規情況下return時會銷毀。
* 但這個$client是異步客戶端在執行connect時swoole引擎底層會增加一次引用計數,因此return時并不會銷毀。
* 該客戶端執行onReceive回調函數時進行了close或者服務器端主動關閉連接觸發onClose,這時底層會減少引用計數,$client才會被銷毀。
## 異步回調程序內存管理
異步回調程序與同步阻塞程序的內存管理方式不同,異步程序是基于回調鏈引用計數實現內存的管理。本文會用一個最簡單的實例講解異步程序的內存管理。
實例程序
~~~
$serv = new Swoole\Http\Server("127.0.0.1", 9502);
$serv->on('Request', function($request, $response) {
$cli = new Swoole\Http\Client('127.0.0.1', 80);
$cli->post('/dump.php', array("key" => 'value'), function ($cli) use ($request, $response) {
$response->end("<h1>{$cli->body}</h1>");
$cli->close();
});
});
$serv->start();
~~~
**onRequest**
* 請求到來這時會觸發onRequest回調函數,可以得到$request和$response對象
在onRequest回調函數中,創建了一個Http\Client,并發起一次POST請求
* 然后onRequest函數結束并返回
* 這時按照正常的PHP函數調用流程,$request和$response對象會被銷毀。但在上述程序中,$request和$response對象被使用了use語法,綁定到了匿名函數上,因此這2個對象的引用計數會被加1。onRequest函數返回時就不會真正銷毀這2個對象了。
$cli對象,是在onRequest函數創建的局部變量,按照正常邏輯$cli對象在onRequest函數退出時也應該被銷毀。但Swoole底層有一個特殊的邏輯,所有異步客戶端對象在發起連接時底層會自動增加一次引用計數,在連接關閉時減少一次引用計數,因此$cli對象也不會銷毀,POST請求中的匿名函數對象也不會銷毀。
**Http響應**
* 創建的$cli對象,接收到來自服務器端的響應,或者連接超時、響應超時,這時會回調指定的匿名函數,調用end向客戶端發送響應
* 回調函數中調用了$cli->close這時切斷連接,$cli的引用計數減一。這時匿名函數退出底層會自動銷毀$cli、$request、$response 3個對象
**多層嵌套**
如果Http\Client的回調函數中調用了其他的異步客戶端,如Swoole\Redis,對象會繼續傳讀引用,形成一個異步調用鏈。當調用鏈的最后一個對象銷毀時會向著調用鏈頭部逐個遞減引用計數,最終銷毀對象。
~~~
$serv = new Swoole\Http\Server("127.0.0.1", 9502);
$serv->on('Request', function($request, $response) {
$cli = new Swoole\Http\Client('127.0.0.1', 80);
//發起連接,$cli 引用計數增加
$cli->post('/dump.php', array("key" => 'value'), function ($cli) use ($request, $response) {
$redis = new Swoole\Redis;
//發起連接,$redis 引用計數增加
$redis->connect('127.0.0.1', 6379, function ($redis, $result) use ($request, $response, $cli) {
$redis->get('test_key', function ($redis, $result) use ($request, $response, $cli) {
$response->end("<h1>{$result}</h1>");
//關閉連接,$cli 引用計數減少
$cli->close();
//關閉連接,$redis 引用計數減少
$redis->close();
});
});
});
});
$serv->start();
~~~
* 這里$response和$request對象被POST匿名函數、Redis->connect匿名函數、Redis->get匿名函數引用,因此需要等到這3個函數執行后,引用計數減少為0,才會真正的銷毀
* $cli和$redis對象在發起TCP連接時,會被Swoole底層增加引用計數。只有$cli->close()和$redis->close被調用,或者遠端服務器關閉連接,觸發$cli->onClose和$redis->onClose,$cli和$redis這2個對象的,引用計數才會減少,函數退出時會銷毀
* POST匿名函數、Redis->connect匿名函數、Redis->get匿名函數,3個對象依附于$cli和$redis對象,當$cli和$redis對象銷毀時,這3個對象也會被銷毀
* POST匿名函數、Redis->connect匿名函數、Redis->get匿名函數,匿名函數銷毀時通過use語法引用的$response和$request對象也會銷毀
- swoole簡介
- swoole功能概述
- 序章
- 開發必讀
- 1 環境搭建
- 1.1 環境搭建
- 1.2 搭建Echo服務器
- 2 初識Swoole
- 2.1 Worker進程
- 2.2 TaskWorker進程
- 2.3 Timer定時器
- 2.4 Process進程
- 2.5 Table內存表
- 2.6 多端口監聽
- 2.7 sendfile文件支持
- 2.8 SSL支持
- 2.9 熱重啟
- 2.10 http_server
- 附錄*server配置
- 附錄*server函數
- 附錄*server屬性
- 附錄*server回調函數
- 附錄*server高級特性
- 心跳檢測
- 3 Swoole協議
- 3.1 EOF協議
- 3.2 固定包頭協議
- 3.3 Http協議
- 3.4 WebSocket協議
- 3.5 MTQQ協議
- 內置http_server
- 內置websocket_server
- Swoole\Redis\Server
- 4 Swoole異步IO
- 4.1 AsyncIO
- 異步文件系統IO
- swoole_async_readfile
- swoole_async_writefile
- swoole_async_read
- swoole_async_write
- 5 swoole異步客戶端
- ws_client
- http_client
- mysql_client
- redis_client
- tcp_client
- http2_client
- 6 swoole協程
- Swoole\Coroutine\Http\Client
- Swoole\Coroutine\MySQL
- Swoole\Coroutine\Redis
- Coroutine\PostgreSQL
- Swoole\Coroutine\Client
- Swoole\Coroutine\Socket
- Swoole\Coroutine\Channel
- Coroutine
- Swoole\Coroutine::create
- Swoole\Coroutine::resume
- Swoole\Coroutine::suspend
- Swoole\Coroutine::sleep
- Coroutine::getaddrinfo
- Coroutine::gethostbyname
- swoole_async_dns_lookup_coro
- Swoole\Coroutine::getuid
- getDefer
- setDefer
- recv
- Coroutine::stats
- Coroutine::fread
- Coroutine::fget
- Coroutine::fwrite
- Coroutine::readFIle
- Coroutine::writeFIle
- Coroutine::exec
- 7 swoole_process
- process::construct
- process::start
- process::name
- process::signal
- process::setaffinity
- process::exit
- process::kill
- process::daemon
- process->exec
- process::wait
- process::alarm
- 8 swoole定時器
- swoole_timer_tick
- swoole_timer_after
- swoole_timer_clear
- 9 swoole_event
- swoole_event_add
- swoole_event_set
- swoole_event_del
- swoole_event_wait
- swoole_event_defer
- swoole_event_write
- swoole_event_exit
- swoole提供的function
- 常見問題
- 客戶端鏈接失敗原因
- 如何設置進程數
- 如何實現異步任務
- 如何選擇swoole三種模式
- php中哪些函數是阻塞的
- 是否可以共用1個redis或mysql連接
- 如何在回調函數中訪問外部的變量
- 為什么不要send完后立即close
- 不同的Server程序實例間如何通信
- MySQL的連接池、異步、斷線重連
- 在php-fpm或apache中使用swoole
- 學習Swoole需要掌握哪些基礎知識
- 在phpinfo中有在php-m中沒有
- 同步阻塞與異步非阻塞選擇
- CURL發送POST請求服務器端超時
- 附錄
- 預定義常量
- 內核參數調優
- php四種回調寫法
- 守護進程程序常用數據結構
- swoole生命周期
- swoole_server中內存管理機制
- 使用jemalloc優化swoole內存分配性能
- Reactor、Worker、Task的關系
- Manager進程
- Swoole的實現
- Reactor線程
- 安裝擴展
- swoole-worker手冊
- swoole相關開源項目
- 寫在后面的話
- 版本更新記錄
- 4.0