【问题标题】:How to detect client disconnection on PHP socket listener?如何在 PHP 套接字侦听器上检测客户端断开连接?
【发布时间】:2011-04-29 02:45:04
【问题描述】:

我一直在测试 PHP 套接字侦听,并遇到了上述问题。我的测试侦听器工作正常,但是如果客户端在没有通知服务器的情况下断开连接,则脚本将进入无限循环,直到新客户端连接。问题似乎出在$ready = socket_select($read, $write = NULL, $except = NULL, $tv_sec = NULL); 线上,因为它应该在这里停止等待连接,而是跳过等待并直接运行循环。

任何指针将不胜感激。

代码:

#!/usr/bin/php -q
<?php

$debug = true;
function e($str) {
    global $debug;
    if($debug) { echo($str . "\n"); }
}

e("Starting...");
error_reporting(1);
ini_set('display_errors', '1');

e("Creating master socket...");
$socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
$max_clients = 10;

e("Setting socket options...");
socket_set_option($socket, SOL_SOCKET, SO_REUSEADDR, 1);
e("Binding socket...");
socket_bind($socket, "192.168.1.38", 20000);
e("Listening...");
socket_listen($socket, $max_clients);

$clients = array('0' => array('socket' => $socket));

while(TRUE) {
    e("Beginning of WHILE");
    $read[0] = $socket;

    for($i=1; $i<count($clients)+1; ++$i) {
        if($clients[$i] != NULL) {
            $read[$i+1] = $clients[$i]['socket'];
        }
    }

    e("Selecting socket...");
    $ready = socket_select($read, $write = NULL, $except = NULL, $tv_sec = NULL);
    e("socket_select returned " . $ready);
    e("If...");
    var_dump($socket);
    var_dump($read);
    if(in_array($socket, $read)) {
        e("If OK");
        for($i=1; $i < $max_clients+1; ++$i) {
            if(!isset($clients[$i])) {
                e("Accepting connection...");
                $clients[$i]['socket'] = socket_accept($socket);

                socket_getpeername($clients[$i]['socket'],$ip);
                e("Peer: " . $ip);
                $clients[$i]['ipaddr'] = $ip;

                socket_write($clients[$i]['socket'], 'Welcome to my Custom Socket Server'."\r\n");
                socket_write($clients[$i]['socket'], 'There are '.(count($clients) - 1).' client(s) connected to this server.'."\r\n");

                echo 'New client connected: ' . $clients[$i]['ipaddr'] .' ';
                break;
            } elseif($i == $max_clients - 1) {
                echo 'Too many Clients connected!'."\r\n";
            }

            if(--$ready <= 0) {
                continue;
            } 
        }
    }

    e("For...");
    for($i=1; $i<$max_clients+1; ++$i) {
        e("In...");
        if(in_array($clients[$i]['socket'], $read)) {
            e("Reading data...");
            $data = @socket_read($clients[$i]['socket'], 1024, PHP_NORMAL_READ);

            if($data === FALSE) {
                unset($clients[$i]);
                echo 'Client disconnected!',"\r\n";
                continue;
            }

            $data = trim($data);

            if(!empty($data)) {
                if($data == 'exit') {
                    socket_write($clients[$i]['socket'], 'Thanks for trying my Custom Socket Server, goodbye.'."\n");
                    echo 'Client ',$i,' is exiting.',"\n";
                    socket_close($clients[$i]['socket']);
                    unset($clients[$i]);
                    continue;
                } 

                for($j=1; $j<$max_clients+1; ++$j) {
                    if(isset($clients[$j]['socket'])) {
                        if(($clients[$j]['socket'] != $clients[$i]['socket']) && ($clients[$j]['socket'] != $socket)) {
                            echo($clients[$i]['ipaddr'] . ' is sending a message!'."\r\n");
                            socket_write($clients[$j]['socket'], '[' . $clients[$i]['ipaddr'] . '] says: ' . $data . "\r\n");
                        }
                    }
                }
                break;
            }
        }
    }
    if($loops == 0) {
        $firstloop = time();
        $loops++;
    } else {
        $loops++;
        if((time() - $firstloop) >= 5 && $loops > 25) {
            /*for($j=1; $j<$max_clients+1; ++$j) {
                if(isset($clients[$j]['socket'])) {
                    if($clients[$j]['socket'] != $socket) {
                        echo('Server is looping, sending keepalive...'."\r\n");
                        if(!socket_write($clients[$j]['socket'], '-KEEPALIVE-' . "\r\n")) {
                            echo 'Client ',$j,' not found, killing...',"\n";
                            socket_close($clients[$j]['socket']);
                            unset($clients[$j]);
                            die("debug");
                        }
                    }
                }
            }*/
            die("Looping started.\n");
        }
    }
}
?>

【问题讨论】:

  • 考虑更好的代码设计
  • @Svisstack 通常我不太关注测试代码,因为如果我开始进一步开发,我会进行完整的结构化重写。
  • 请问我有同样的问题,你怎么解决的?你做了 if($data === FALSE) ,我的问题是虽然客户端断开连接(当我断开互联网时)但我没有任何 chatnged_sockets ? :(

标签: sockets php


【解决方案1】:

找出问题所在,我在“客户端断开连接”块中丢失了 socket_close()。所以正确的块是:

if($data === FALSE) {
    socket_close($clients[$i]['socket']);
    unset($clients[$i]);
    echo 'Client disconnected!',"\r\n";
    continue;
}

【讨论】:

    【解决方案2】:

    使用get_last_error() 进行检查。

    【讨论】:

    • 104:连接被对等方重置,但我如何正确关闭正确的套接字?
    • 这不是操作码吗?连接关闭?尝试对此进行模拟并检查在这种情况下此函数返回的操作码并为其定义。
    • 我在我的服务器上运行脚本并使用 SocketTest 连接到它。当我断开客户端连接时,服务器会给出错误代码 104。使用的代码:$errorcode = socket_last_error(); $errormsg=socket_strerror($errorcode); die("Error: (".$errorcode.") ".$errormsg."\n");
    • 问题不在于找出错误是什么,而是在错误发生时如何阻止脚本循环。
    猜你喜欢
    • 2013-12-10
    • 2014-04-14
    • 2013-02-11
    • 2010-10-17
    • 1970-01-01
    相关资源
    最近更新 更多