服务器中判断客户端socket断开连接的方法

1, 如果服务端的Socket比客户端的Socket先关闭,会导致客户端出现TIME_WAIT状态,占用系统资源。

所以,必须等客户端先关闭Socket后,服务器端再关闭Socket才能避免TIME_WAIT状态的出现。

2, 在linux下写socket的程序的时候,如果尝试send到一个disconnected socket上,就会让底层抛出一个SIGPIPE信号。

client端通过 pipe 发送信息到server端后,就关闭client端, 这时server端,返回信息给 client 端时就产生Broken pipe 信号了。

    当服务器close一个连接时,若client端接着发数据。根据TCP协议的规定,会收到一个RST响应,client再往这个服务器发送数据时,系统会发出一个SIGPIPE信号给进程,告诉进程这个连接已经断开了,不要再写了。
    根据信号的默认处理规则SIGPIPE信号的默认执行动作是terminate(终止、退出),所以client会退出。若不想客户端退出可以把SIGPIPE设为SIG_IGN

    如:    signal(SIGPIPE,SIG_IGN);
    这时SIGPIPE交给了系统处理。

 

 

    这个信号的缺省处理方法是退出进程,大多数时候这都不是我们期望的。因此我们需要重载这个信号的处理方法。调用以  下代码,即可安全的屏蔽SIGPIPE:
    struct sigaction sa;
    sa.sa_handler = SIG_IGN;
    sigaction( SIGPIPE, &sa, 0 );

 


  服务器采用了fork的话,要收集垃圾进程,防止僵尸进程的产生,可以这样处理:
  signal(SIGCHLD,SIG_IGN); 交给系统init去回收。
   这里子进程就不会产生僵尸进程了。

 

判断连接断开的方法

法一:

当recv()返回值小于等于0时,socket连接断开。但是还需要判断 errno是否等于 EINTR,如果errno == EINTR 则说明recv函数是由于程序接收到信号后返回的,socket连接还是正常的,不应close掉socket连接。

 

法二:

  struct tcp_info info; 
  int len=sizeof(info); 
  getsockopt(sock, IPPROTO_TCP, TCP_INFO, &info, (socklen_t *)&len); 
  if((info.tcpi_state==TCP_ESTABLISHED))  则说明未断开  else 断开

 

法三:

若使用了select等系统函数,若远端断开,则select返回1,recv返回0则断开。其他注意事项同法一。

 

法四:

int keepAlive = 1; // 开启keepalive属性
int keepIdle = 60; // 如该连接在60秒内没有任何数据往来,则进行探测 
int keepInterval = 5; // 探测时发包的时间间隔为5 秒
int keepCount = 3; // 探测尝试的次数.如果第1次探测包就收到响应了,则后2次的不再发.

setsockopt(rs, SOL_SOCKET, SO_KEEPALIVE, (void *)&keepAlive, sizeof(keepAlive));
setsockopt(rs, SOL_TCP, TCP_KEEPIDLE, (void*)&keepIdle, sizeof(keepIdle));
setsockopt(rs, SOL_TCP, TCP_KEEPINTVL, (void *)&keepInterval, sizeof(keepInterval));
setsockopt(rs, SOL_TCP, TCP_KEEPCNT, (void *)&keepCount, sizeof(keepCount));

设置后,若断开,则在使用该socket读写时立即失败,并返回ETIMEDOUT错误

 

法五:

自己实现一个心跳检测,一定时间内未收到自定义的心跳包则标记为已断开。

另外一网摘,方法如下:

判断客户端Socket的关闭

最近试验发现,当客户端Socket关闭时,服务端的Socket会接收到0字节的通知。

private int Receive(StringBuilder sb)

{

    int read = 0, total = 0;

    if (_Client != null)

    {

        try

        {

            byte[] bytes = new byte[SIZE];

            int available = _Client.Available;

            do

            {

                read = _Client.Receive(bytes);//如果客户端Socket关闭,_Client会接受到read=0

                total += read;

                if (read > 0)

                    sb.Append(_Server.DefaultEncoding.GetString(bytes, 0, read));

 

            } while (read > 0 && total < available);

        }

        catch (SocketException)

        {

            CloseSocket();

        }

    }

    if (_Server.TraceInConsole && total > 0)

    {

        Console.WriteLine("Receive:" + total + "======================================");

        Console.WriteLine(sb.ToString());

    }

    return total;

}

利用0字节接收条件判断客户端Socket的关闭,开始执行服务端Socket关闭代码。

private void ThreadHandler()

{

    if (_Server.TraceInConsole)

        Console.WriteLine("Begin HttpRequest...");

    try

    {

        while (true)

        {

            StringBuilder sb = new StringBuilder();

            int receive = Receive(sb);

            if (receive > 0)

            {

                _Server.ReadRequest(this, sb.ToString());

                _Server.Response(this);

                _Server.ResponseFinished(this);

            }

            else

            {

                TryCloseSocket();

            }

            if (_Client == null)

                break;

        }

    }

    catch (Exception ex)

    {

        if (_Server.TraceInConsole)

            Console.WriteLine(ex.Message);

    }

    if (_Server.TraceInConsole)

        Console.WriteLine("End HttpRequest.");

}

服务端Socket的关闭

如果直接调用Socket的Close方法会关闭得太快,可能导致客户端TIME_WAIT现象;而Thead.Sleep延时再调用Socket的Close方法也不理想。应该采用尝试向客户端发送数据,然后利用异常来关闭Socket,方法如下。

private void TryCloseSocket()

{

    try

    {

        while (true)

        {

            Thread.Sleep(1500);

            Send(HttpServer.BYTES_CRLF); //发送自定义的字节,如果客户端关闭出现SocketException,然后关闭服务端Socket

            if (_Client == null)

                break;

        }

    }

    catch (SocketException)

    {

        CloseSocket();

    }

}

 

private void CloseSocket()

{

    if (_Client != null)

    {

        _Client.Shutdown(SocketShutdown.Both);

        _Client.Close();

        _Client = null;

        if (_Server.TraceInConsole)

        {

            Console.WriteLine("Close socket.");

        }

    }

}

 

  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 判断客户端是否断开连接,可以通过检测TCP连接的状态来实现。如果TCP连接状态为CLOSED或RESET,表示客户端已经断开连接。可以在服务器端使用socket编程的getsockopt函数来获取TCP连接的状态信息。 ### 回答2: TCP服务器需要判断客户端是否断开连接,因为TCP协议提供的是可靠的连接服务,当客户端意外关闭连接或出现异常时,服务器需要知道这个事件的发生,以便及时做出处理,例如清理相应的资源或告知其他客户端。 一般来说,TCP服务器可以通过两种方式判断客户端是否断开连接: 1. 使用心跳包机制 心跳包是用于检测TCP连接是否存活的一种机制,服务器客户端定时互相发送一个信号包,如果一方收不到对方的回应,则说明连接已经断开服务器可以定时向客户端发送心跳包,如果一定时间内没有收到客户端的回应,则可以判断客户端已经断开连接。 2. 使用select或epoll函数 在TCP服务器,可以使用select或epoll函数来检测socket的可读事件。当客户端断开连接时,服务器端的socket会接收到一个可读事件(即收到一个EOF),此时select或epoll函数会返回一个readable事件,服务器端可以据此判断客户端已经断开连接。 以上两种方法都可以判断客户端是否断开连接,但是心跳包机制需要额外的代码实现,而使用select或epoll函数则可以避免这个问题。不过,需要注意的是,当客户端网络异常或出现故障而未能及时关闭连接时,以上两种方法可能会出现误判,因此需要谨慎使用。 ### 回答3: 当TCP服务器客户端建立连接时,会通过三次握手来确认连接。而当客户端主动断开连接时,会向服务器发送FIN码,表示不再发送数据,但仍然可以接收来自服务器的数据。如图所示: ![TCP断开连接握手过程](https://www.linuxidc.com/upload/2009_03/090310153217131.jpg) 当服务器收到一个FIN码时,应该如何判断客户端是否已经断开连接呢?以下是几种可能的方法: 1.使用select函数:select函数是一个非常常用的函数,可以用于监控多个文件描述符的状态。当select函数发现一个套接字处于可读状态时,它表明这个套接字已经准备好接收数据了。而当select函数返回0时,表明所有监控的文件描述符都没有事件发生。在服务器,可以将所有连接的套接字加入到select函数的监控列表,在处理可读事件时,当套接字读取到EOF时,表明客户端已经断开连接了。 2.使用SO_KEEPALIVE选项:SO_KEEPALIVE选项可以在TCP套接字上启用周期性的探测机制,用于检测连接是否仍然存活。当启用SO_KEEPALIVE选项后,如果客户端在一段时间内没有发送任何数据,服务器会周期性地向客户端发送探测数据包,如果客户端没有回应,则认为客户端已经断开连接。 3.使用心跳包:心跳包是一种特殊的数据包,用于检测连接是否仍然存活。在服务器,可以定时向客户端发送心跳包,当客户端回应心跳包时,表明客户端仍然存活,否则表明客户端已经断开连接。 总之,在TCP服务器判断客户端是否已经断开连接,需要根据具体情况采取不同的方法,选择合适的方法可以提高服务器性能,降低服务器负载。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值