- 我在使用swoole开发即时通讯软件的过程中,发现swoole的worder进程中,使用redis操作会导致内存泄露,并且在不同的操作系统中,内存泄露的大小不同。
- 测试代码如下:
<?php
$server = new swoole_websocket_server("0.0.0.0", 4096);
$redis = new \Redis();
$redis->connect('127.0.0.1',20002);
$server->set(array(
'worker_num' => 1,
'daemonize' => 0,
'max_request' => 0,
'task_worker_num' => 1
));
$server->on('workerStart', function ($ser, $worker_id) {
});
$server->on('task', function($serv, $task_id, $from_id, $msgs){
});
$server->on('finish', function($serv, $task_id, $data){
// print_r($data);
});
$server->on('open', function (swoole_websocket_server $server, $request) {
// echo "server: handshake success with fd{$request->fd}\n";
});
$server->on('message', function (swoole_websocket_server $server, $frame) use($redis) {
echo "receive from {$frame->fd}:{$frame->data},opcode:{$frame->opcode},fin:{$frame->finish}\n";
$redis->incr('test');
$server->push($frame->fd, "this is server");
});
$server->on('close', function ($ser, $fd) {
// echo "client {$fd} closed\n";
});
$server->start();
- 测试过程
swoole版本:4.2.12
redis:3.1.15
测试工具:jemeter
测试线程:2000
循环次数:5
监控:top命令,或者docker stats命令监控内存使用情况随着压测的进行,内存不断累积,即时中断链接,内存也没有释放,反复压测,会导致worker进程挂掉重启。
- 结论发现
经过测试,我发现,当操作系统是centos7.2的时候,内存激增幅度很大,每一轮压测会导致上百M的累积,而当操作系统是 alpine linux 3.4的时候,内存增幅相对会小很多,没一轮压测只会导致5M左右的累积。 - 疑问
有其他同学碰到这个问题,有没有解决方案?
https://wiki.swoole.com/wiki/...
这个参数的主要作用是解决PHP进程内存溢出问题。PHP应用程序有缓慢的内存泄漏,但无法定位到具体原因、无法解决,可以通过设置max_request解决。