【问题标题】:Socket server receiving data at buffer size套接字服务器以缓冲区大小接收数据
【发布时间】:2020-06-15 19:19:58
【问题描述】:

我一个接一个地发送不同大小的数据包,如何以我发送的大小分别接收数据包,而不是在缓冲区中累积。似乎现在服务器会添加到缓冲区,直到它填满它,然后我可以处理它们。

例子:

缓冲区大小:84。

从客户端发送:84 字节、76 字节、76 字节、80 字节

在服务器中接收:84 字节、84 字节、84 字节、64 字节。

我希望在发送它们时收到它们。有可能吗?

    int port = stoi(getConfig("server_port"));
    std::string ipAddress = getConfig("ip_address");

    // Create a socket
    int listening = socket(AF_INET, SOCK_STREAM, 0);
    if (listening < 0){
        std::cerr << "Can't create a socket!" << endl;
        Logger("Can't create a socket!");
        exit(-1);
    }
    std::cout << "The socket server was created successfully." << endl;

    // Bind the socket to a IP / port
    sockaddr_in hint;
    hint.sin_family = AF_INET;
    hint.sin_port = htons(port);
    inet_pton(AF_INET, ipAddress.c_str(), &hint.sin_addr);

    if (bind(listening, (sockaddr*)&hint, sizeof(hint)) < 0){
        cerr << "Can't bind to IP/port!" << endl;
        Logger("Can't bind to IP/port!");
        exit(-1);
    }

    // Mark the socket for listening in
    if (listen(listening, SOMAXCONN) < 0){
        cerr << "Can't listen!" << endl;
        Logger("Can't listen!");
        exit(-1);
    }

    // Accept a call
    sockaddr_in client;
    socklen_t clientSize = sizeof(client);
    char host[NI_MAXHOST];
    char svc[NI_MAXSERV];

    while(true){
        int clientSoket = accept(listening, (sockaddr*)&client, &clientSize);
        if(clientSoket < 0){
            cerr << "Problem with client connecting!" << endl;
            Logger("Problem with client connecting!");
            break;
        }
        cout << "The client whas conected successfully." << endl;

        memset(host, 0, NI_MAXHOST);
        memset(svc, 0, NI_MAXSERV);

        int result = getnameinfo((sockaddr*)&client, clientSize, host, NI_MAXHOST, svc, NI_MAXSERV, 0);
        if(result == 0) {
            cout << host << " connected on " << svc << endl;
        } else {
            inet_ntop(AF_INET, &client.sin_addr, host, NI_MAXHOST);
            cout << host << " connected on " << ntohs(client.sin_port) << endl;
        }

        // While receiving
        char buff[84];
        while(true){
            // Clear the buffer
            memset(buff, 0, sizeof(buff));

            // Wait for a message
            int bytesRecv = recv(clientSoket, buff, sizeof(buff), 0);
            if(bytesRecv < 0){
                cerr << "There was a connection issue!" << endl;
                Logger("There was a connection issue!");
                break;
            }

            if(bytesRecv == 0){
                cout << "The client disconnected." << endl;
                Logger("The client disconnected");
                break;
            }

            cout << "bytesRecv: " << bytesRecv << endl;
       }

       // Close the socket
       close(clientSoket);
    }

【问题讨论】:

标签: c++ tcpserver


【解决方案1】:

不,流套接字不能那样工作。

流套接字是一个非结构化的字节流,没有任何结构。在这方面,它与普通文件没有什么不同。如果您将不同大小的记录写入一个普通文件,并且现在准备回读它们,那么您希望如何读取可变大小的记录?

无论你在这里给出什么答案,同样的答案适用于套接字,另外一个额外的转折是,套接字上的 read() 不能保证你将阅读多少,除了它会小于或等于size 参数到read()。这是您从read() 获得的唯一保修。

如果发送者调用write() 两次(顺便说一句,套接字也不能保证无论你想多少write,都会写入多少,write 也可以返回小于或等于它的size 参数,这取决于你的代码如何处理它),一旦写入 76 个字节,第二次写入 84,read()ing (假设足够大的缓冲区大小)可以在初始读取时读取 1 到 160 字节之间的任意字节数。

如果您希望实现一些正式的结构、某种记录,则由您自己决定如何在这些约束条件下实现它。也许通过发送每条记录的大小(以字节为单位),然后是记录本身。或者做任何你想做的事。请记住,您无法保证个人read() 会返回多少。例如,如果您首先发送记录计数,作为四字节值。您最初的read() 可能会返回一个、两个或三个字节。您的代码必须准备好应对任何可能发生的情况。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2011-11-15
    • 1970-01-01
    • 1970-01-01
    • 2012-04-23
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2017-10-02
    相关资源
    最近更新 更多