接受的套接字上出现“错误文件描述符”错误

Toc*_*ave 3 c sockets windows networking

我正在尝试进入网络编程并一直在遵循套接字指南。我用 C 语言编写了一个简单的 Windows TCP 服务器和客户端,但似乎无法让它们工作。客户端连接后,服务器会抛出“错误文件描述符”错误并关闭。我不明白刚刚创建的文件描述符怎么accept()可能是坏的。我对此进行了研究,只发现有人在阅读之前关闭了该文件,而我却没有。

这是代码:

服务器

#include <stdio.h>
#include <stdlib.h>
#include <winsock2.h>

int main(int argc, char *argv[])
{
    WSADATA WSAData;
    WSAStartup(MAKEWORD(2,0), &WSAData);

    SOCKET sock, newsock;
    int portno, clilen;
    char buffer[256];
    SOCKADDR_IN serv_addr, cli_addr;
    int n;

    sock = socket(AF_INET, SOCK_STREAM, 0);

    if (sock < 0)
    {
        perror("Error opening socket");
        exit(1);
    }

    memset((char *) &serv_addr, 0, sizeof(serv_addr));
    portno = 5001;

    serv_addr.sin_family = AF_INET;
    serv_addr.sin_addr.s_addr = INADDR_ANY;
    serv_addr.sin_port = htons(portno);

    if (bind(sock, (SOCKADDR *) &serv_addr, sizeof(serv_addr)) < 0)
    {
        perror("Eror on binding");
        exit(1);
    }

    listen(sock, 5);
    clilen = sizeof(cli_addr);

    newsock = accept(sock, (SOCKADDR *) &cli_addr, &clilen);
    if (newsock < 0)
    {
        perror("Error on accept");
        exit(1);
    }

    memset(buffer, 0, 256);
    n = read(newsock, buffer, 255);

    if (n < 0)
    {
        perror("Error reading from socket");
        exit(1);
    }

    printf("Here is the message : %s\n", buffer);

    n = write(newsock, "I got your message", 18);
    if (n < 0) 
    {
        perror("Error writing to socket");
        exit(1);
    }

    WSACleanup();
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

客户

#include <stdio.h>
#include <stdlib.h>
#include <winsock2.h>

int main(int argc, char *argv[])
{
    WSADATA WSAData;
    WSAStartup(MAKEWORD(2,0), &WSAData);

    SOCKET sock;
    int portno, n;
    SOCKADDR_IN serv_addr;
    HOSTENT *server;
    char buffer[256];

    if (argc < 3)
    {
        fprintf(stderr, "Usage : %s hostname port\n", argv[0]);
        exit(0);
    }

    portno = atoi(argv[2]);
    sock = socket(AF_INET, SOCK_STREAM, 0);
    if (sock < 0)
    {
        perror("Error opening socket\n");
        exit(1);
    }

    server = gethostbyname(argv[1]);

    if (server == NULL)
    {
        perror("Error, no such host");
        exit(1);
    }

    memset((char *) &serv_addr, 0, sizeof(serv_addr));
    serv_addr.sin_family = AF_INET;
    memcpy((char *) &serv_addr.sin_addr.s_addr, (char *) server->h_addr, server->h_length);
    serv_addr.sin_port = htons(portno);

    if (connect(sock, (SOCKADDR *) &serv_addr, sizeof(serv_addr)) < 0)
    {
        perror("Error connecting to server");
        exit(1);
    }

    printf("Please enter your message : ");
    memset(buffer, 0, 256);
    fgets(buffer, 255, stdin);

    n = write(sock, buffer, strlen(buffer));
    if (n < 0)
    {
        perror("Error writing to socket");
        exit(1);
    }

    memset(buffer, 0, 256);
    n = read(sock, buffer, 255);

    if (n < 0)
    {
        perror("Error reading from socket");
        exit(1);
    }

    printf("%s\n", buffer);

    WSACleanup();
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

...这是以防万一的输出:

C:\...myfolder>server
*here it does nothing until I start the client with "client 127.0.0.1 5001" in another window*
Error reading from socket: Bad file descriptor
Run Code Online (Sandbox Code Playgroud)

你看出我的代码有什么问题了吗?在此先感谢您的帮助!

小智 5

我会将其放在注释中,但尝试使用recv()函数而不是 read()。这是因为 read() 不会调用 Socket API,因此对于 Windows 而言,文件描述符无效。

同样,在您的客户端中,您应该使用send()而不是 write()。

  • *任何东西*真的是 Windows 上的文件描述符吗?还是只是一堆假装是文件描述符的东西? (2认同)
  • 回到 TCP/IP 网络包含在 Windows 中之前的时代,winsock 是第三方库,不可能将这些新奇的“套接字”对象与现有的文件句柄系统集成,因此创建了文件描述符通过winsock 函数在winsock 库之外没有任何意义,这意味着您必须使用recv 而不是read。不知何故,当winsock成为操作系统的一部分时,这种情况并没有得到改善。 (2认同)