我有一个用 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
获取二进制数据的长度时,有两个可能的问题:
数据中间有一个零字节。这将导致
strlen
提前终止并返回错误的长度。数据中没有 零字节。这将导致
strlen
越过缓冲区末尾寻找零字节,从而导致未定义的行为。
关于c++ - 如何通过linux套接字发送图像数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32426269/