【问题标题】:Converting the host byte order to network byte order sometimes results in odd results将主机字节顺序转换为网络字节顺序有时会导致奇怪的结果
【发布时间】:2015-06-28 08:52:40
【问题描述】:

我目前正在用 C++ 编写一个供 android 客户端使用的服务器。基本上我使用 googles 协议缓冲区进行数据传输,在发送 protobuf 包之前,我以网络字节顺序发送包大小。这是我发送一个包裹的功能:

bool Client::sendOnePackage(const std::string & data) {
    int32_t packageSize = data.length();
    packageSize = htonl(packageSize);

    char packageSizeBuffer[sizeof (int32_t)];
    memcpy(packageSizeBuffer, &packageSize, sizeof (int32_t));

    if (send(con.sockfd, packageSizeBuffer, sizeof (int32_t), 0) == sizeof (int32_t)) {
        if (send(con.sockfd, data.c_str(), packageSize, 0) == packageSize) {
            return true;
        }
    }
    return false;
}

然后,android 部分将使用以下方法处理将发送的整数转换为主机字节顺序的包:

int answerLength = ByteBuffer.wrap(answerLengthData).getInt();

现在的问题:当我通过网络发送一个大小为 8 的包时,它工作得很好,android 接收到字节数组 0,0,0,8(总是 4 个字节)。但是当我通过网络发送值 6 时,android 会收到奇数字节数组 0,114,0,0,这会导致异常大的包大小 (7471104)。

我在这里遗漏了什么吗?如果您需要更多信息或更多代码,请索取,然后发布。

【问题讨论】:

  • 与此问题无关,但不需要临时字符缓冲区,直接使用整数变量即可。例如。 send(con.sockfd, reinterpret_cast<char*>(&packageSize), sizeof(packageSize), 0)
  • 您的变量 packageSize 被转换为网络字节顺序,然后您尝试将其用作长度。

标签: java android c++ sockets networking


【解决方案1】:

你用htonl()的结果覆盖packageSize的值:

packageSize = htonl(packageSize);

然后你将它用作传递给send()的包大小:

send(con.sockfd, data.c_str(), packageSize, 0)

将相关值保存在不同的变量中。并且不要使用临时字符缓冲区:

bool Client::sendOnePackage(const std::string & data) {
    int32_t packageSize = data.length();
    int32_t conv_size = htonl(packageSize);

    if (send(con.sockfd, reinterpret_cast<char*>(&conv_size), sizeof (int32_t), 0) == sizeof (int32_t)) {
        if (send(con.sockfd, data.c_str(), packageSize, 0) == packageSize) { //use original size
            return true;
        }
    }
    return false;
}

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2022-01-22
    • 2013-10-23
    • 1970-01-01
    • 2023-03-11
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多