c++ - 如何通过linux套接字发送图像数据

标签 c++ linux sockets

我有一个用 C++ 编写的相对简单的 Web 服务器。它适用于提供文本/html 页面,但它的编写方式似乎无法发送二进制数据,我真的需要能够发送图像。

我一直在搜索和搜索,但找不到针对这个问题的特定答案,它是用真正的 C++ 编写的(fstream 而不是使用文件指针等),虽然这种事情必然是低级的,但很可能需要处理 C 样式数组中的字节我希望代码尽可能像 C++。

我尝试了几种方法,这是我目前拥有的:

int sendFile(const Server* serv, const ssocks::Response& response, int fd)
{
// some other stuff to do with headers etc. ........ then:

    // open file
    std::ifstream fileHandle;
    fileHandle.open(serv->mBase + WWW_D + resource.c_str(), std::ios::binary);

    if(!fileHandle.is_open())
    {            
        // error handling code
        return -1;
    }

    // send file
    ssize_t buffer_size = 2048;
    char buffer[buffer_size];

    while(!fileHandle.eof())
    {
        fileHandle.read(buffer, buffer_size);

        status = serv->mSock.doSend(buffer, fd);
        if (status == -1)
        {
            std::cerr << "Error: socket error, sending file\n";
            return -1;
        }
    }
    return 0
}

然后在其他地方:

int TcpSocket::doSend(const char* message, int fd) const
{
    if (fd == 0)
    {
         fd = mFiledes;
    }

    ssize_t bytesSent = send(fd, message, strlen(message), 0);
    if (bytesSent < 1)
    {
        return -1;
    }

    return 0;
}

正如我所说,问题是当客户端请求图像时它不会工作。我进入 std::cerr "Error: socket error sending file"

编辑:我使用我接受的答案中的建议让它工作。为了完整性和帮助那些找到这篇文章的人,我还发布了最终的工作代码。

为了发送,我决定使用 std::vector 而不是 char 数组。主要是因为我觉得它是一种更 C++ 的方法,它清楚地表明数据不是字符串。这可能不是必需的,而是品味问题。然后我计算了为流读取的字节数并将其传递给发送函数,如下所示:

// send file
std::vector<char> buffer(SEND_BUFFER);
while(!fileHandle.eof())
{
    fileHandle.read(&buffer[0], SEND_BUFFER);
    status = serv->mSock.doSend(&buffer[0], fd, fileHandle.gcount());
    if (status == -1)
    {
        std::cerr << "Error: socket error, sending file\n";
        return -1;
    }
}

然后实际的发送函数被改编成这样:

int TcpSocket::doSend(const char* message, int fd, size_t size) const
{
    if (fd == 0)
    {
        fd = mFiledes;
    }

    ssize_t bytesSent = send(fd, message, size, 0);
    if (bytesSent < 1)
    {
        return -1;
    }

    return 0;
}

最佳答案

您应该更改的第一件事是 while (!fileHandle.eof()) 循环,因为它不会像您期望的那样工作,实际上它会迭代太多次,因为eof 标志只有在您尝试从文件末尾读取之后才会设置。相反,例如while (fileHandle.read(...)).

您应该做的第二件事是检查实际从文件中读取了多少字节,并且只发送该字节数。

最后,您读取的是二进制数据,而不是文本,因此您不能对从文件中读取的数据使用strlen


对二进制文件问题的一点解释:正如你应该知道的那样,C 风格的字符串(你使用 strlen 来获取长度的字符串)以零字符 结束'\0'(简而言之,零字节)。随机二进制数据可以在其内部的任何位置包含大量零字节,并且它是一个有效字节并且没有任何特殊含义。

当您使用strlen 获取二进制数据的长度时,有两个可能的问题:

  1. 数据中间有一个零字节。这将导致 strlen 提前终止并返回错误的长度。

  2. 数据中没有 零字节。这将导致 strlen 越过缓冲区末尾寻找零字节,从而导致未定义的行为

关于c++ - 如何通过linux套接字发送图像数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32426269/

相关文章:

sockets - 为什么 SCTP 不需要 TIME_WAIT 状态?

python - 系统服务不执行通知发送

c++ - 具有重载的矩阵。不能使用多个运算符

c++ - 我将如何优化这段代码?

c++ - 未知类型名称字符串 C++

c - UDP:为两个不同的多播流监听相同的端口

linux - 命令已退出,状态为 1 : pager -s

sockets - Lua socket.http sink 参数

android, libaums : java. io.IOException: 无法写入设备

c++ - glClearBufferfv - 未在此范围内声明?