C服务器套接字关闭但客户端套接字仍然可以发送另外两个包

Posted

技术标签:

【中文标题】C服务器套接字关闭但客户端套接字仍然可以发送另外两个包【英文标题】:C server socket closed but client socket still able send two more packages 【发布时间】:2018-03-23 20:11:12 【问题描述】:

我正在 c 上学习套接字。我有一个客户端和一个服务器,当服务器关闭套接字时,客户端仍然能够接收并发送到服务器另外两个包,然后再发送一个 SIGPIPE 信号。我不知道为什么。有没有人可以帮忙啊~

因为文档说如果sendrecv 有错误,那么它们将返回-1。但这在我这里从来没有发生过。

客户端

#include    <stdio.h>
#include    <stdlib.h>
#include    <string.h>
#include    <unistd.h>
#include    <signal.h>
#include    <sys/types.h>
#include    <sys/socket.h>
#include    <netinet/in.h>
#include    <arpa/inet.h>

#define BUFFERSIZE  255
#define MAXLENG     96
#define true        1
#define false       0

void exitp();

int main(int argc, char *argv[])

    signal(SIGPIPE, exitp);
    // Check if the argument is match the requirement
    if (argc != 3) 
        fprintf(stderr, "Useage Error, should be follow by ip, and port\n" );
        exit(1);
    

    // Create the socket for the client, if the fd for the socket == -1, it means
    // it created fail
    int skfd = socket(AF_INET, SOCK_STREAM, 0);
    if (skfd  < 0)
        fprintf(stderr, "Create socket failed\n" );
        exit(1);
    


    int PORT = atoi(argv[2]);

    // Set up server argument
    struct sockaddr_in server_addr;
    memset(&server_addr, '0', sizeof(server_addr)); // addr for bin
    server_addr.sin_family = AF_INET;
    server_addr.sin_port = htons(PORT);

    // ip not vaild
    if (inet_pton(AF_INET, argv[1], &server_addr.sin_addr) < 1)
        fprintf(stderr, "IP address not correct\n" );
        exit(1);
    

    // create the connection
    if (connect(skfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) == -1) // Connect to server
        fprintf(stderr, "Connection Fail\n" );
        exit(1);
    

    char sentbuff[255], recvbuffer[255], input[255], request[100], concelbuffer[255];


    // Send the request to the server
    int nn;
    int size = send(skfd, request, sizeof(request),0);
    while(1)
        if (recv(skfd, &recvbuffer,sizeof(recvbuffer),0) == -1)
            printf("Server Closed\n");
            break;
        
        printf("%s",recvbuffer);
        fgets(sentbuff, 255, stdin);
        nn=send(skfd, &sentbuff, sizeof(sentbuff), 0);
        if (nn == -1)
        printf("Server Closed\n");
        break;
        
    

    return 0;



void exitp()
    printf("%s\n","Server Closed" );
    exit(0);

服务器端我使用shutdownclose 作为acceptfd

【问题讨论】:

我不知道如何解释你的观察,只是说客户端正在猛击服务器,服务器可能需要一些时间才能关闭。我要做的是在消息协议中添加一个关闭命令,这样它就不会是意外事件,并且可以比对错误条件做出反应更优雅地处理。 【参考方案1】:

send() 只是将数据放入内核套接字缓冲区,它不会等待数据传输或服务器确认收到。

在数据被传输并且服务器通过发送RST 段拒绝它之前,您不会得到SIGPIPE

之所以这样工作,是因为 TCP 连接的每个方向都是独立处理的。当服务器关闭套接字时,它会发送一个FIN 段。这只是告诉客户端服务器发送数据完成,并不意味着服务器无法接收数据。 TCP 协议中没有任何内容允许服务器将此通知客户端。因此,找出它不再接受任何数据的唯一方法是当客户端收到 RST 响应时。

通知客户端他们不应再发送任何内容通常在应用程序协议中完成,因为它在 TCP 中不可用。

【讨论】:

以上是关于C服务器套接字关闭但客户端套接字仍然可以发送另外两个包的主要内容,如果未能解决你的问题,请参考以下文章

从客户端C发送套接字,服务器Python问题

测试套接字连接的好工具? [关闭]

通过C中的套接字发送IP地址

C 套接字:Echo 服务器错误回复

JAVA套接字在接受数据之前关闭

在 C++ 中通过 tcp 套接字发送结构