swoole内使用redis操作,内存泄露

  1. 我在使用swoole开发即时通讯软件的过程中,发现swoole的worder进程中,使用redis操作会导致内存泄露,并且在不同的操作系统中,内存泄露的大小不同。
  2. 测试代码如下:
<?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();
  1. 测试过程
    swoole版本:4.2.12
    redis:3.1.15
    测试工具:jemeter
    测试线程:2000
    循环次数:5
    监控:top命令,或者docker stats命令监控内存使用情况

    随着压测的进行,内存不断累积,即时中断链接,内存也没有释放,反复压测,会导致worker进程挂掉重启。

  2. 结论发现
    经过测试,我发现,当操作系统是centos7.2的时候,内存激增幅度很大,每一轮压测会导致上百M的累积,而当操作系统是 alpine linux 3.4的时候,内存增幅相对会小很多,没一轮压测只会导致5M左右的累积。
  3. 疑问
    有其他同学碰到这个问题,有没有解决方案?
阅读 4.8k
3 个回答

https://wiki.swoole.com/wiki/...

这个参数的主要作用是解决PHP进程内存溢出问题。PHP应用程序有缓慢的内存泄漏,但无法定位到具体原因、无法解决,可以通过设置max_request解决。

你的redis连接有问题
你应该看下文档
https://wiki.swoole.com/wiki/...

redis连接应该是在workerStart里面进行的,这样就可以设置任务数量来重新进行连接来解决内存的问题

可以写协程的redis连接池,这样也可以避免内存溢出的问题

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题