gpt4 book ai didi

c - 用 C 写一个网络服务器

转载 作者:可可西里 更新时间:2023-11-01 16:18:44 24 4
gpt4 key购买 nike

我正在尝试用 C 语言编写一个简单的 Web 服务器。截至目前,我可以完整地接收连接和消息。但是,按照HTTP/1.0协议(protocol),我希望在遇到“\r\n\r\n”序列时,能够将信息回传给客户端。但是,当使用 Telnet 测试我的服务器时,当我输入“\r\n\r\n”时,服务器什么都不做,直到我在客户端上点击“^]”。我针对 Apache 进行了测试,Apache 没有这个问题。所以我希望获得一些关于如何模仿 Apache 行为的信息。下面添加了我的代码,但请记住,我还没有完成任何工作,也没有实现大量错误检查。谢谢!

main(){
int sock_fd = 0;
int client_fd = 0;
struct sockaddr_in socket_struct;
/*Creates the socket*/
if ((sock_fd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
{
fprintf(stderr, "%s: %s\n", getprogname(), strerror(errno));
exit(EXIT_FAILURE);
}/*Ends the socket creation*/

/*Populates the socket address structure*/
socket_struct.sin_family = AF_INET;
socket_struct.sin_addr.s_addr=INADDR_ANY;
socket_struct.sin_port =htons(port);

if (bind(sock_fd, (struct sockaddr*) &socket_struct, sizeof(socket_struct)) < 0)
{
fprintf(stderr, "%s: %s\n", getprogname(), strerror(errno));
exit(EXIT_FAILURE);
}//Ends the binding.

if (listen(sock_fd, 5) <0)
{
fprintf(stderr, "%s: %s\n", getprogname(), strerror(errno));
exit(EXIT_FAILURE);
}//Ends the listening function

if ( (client_fd = accept(sock_fd, NULL, NULL)) <0)
{
fprintf(stderr, "%s: %s\n", getprogname(), strerror(errno));
exit(EXIT_FAILURE);
}//Ends the accepting.
while ( (size = read(client_fd, msg, 1024)) > 0)
{
//size = recv(client_fd, msg, 1024, MSG_PEEK|MSG_WAITALL);
if ( (msg[size-4] == 13) && (msg[size-3] == 10)&&(msg[size-2] == 13) && (msg[size-1] == 10) )
{
char* buffer = (char *)malloc(sizeof("The msg was: ")+ sizeof(msg));
sprintf(buffer, "The msg was: %s", msg);
send(client_fd, buffer, sizeof("The msg was: ")+ sizeof(msg), MSG_OOB);
}

}//ends the while loop for receiving data
close(client_fd);
}

最佳答案

首先,这一行行不通:

while ( (size = read(client_fd, msg, 1024)) > 0)

这将接收消息 block ,并且您收到的每个 block 都会覆盖最后一个 block 。做这样的事情:

char msg[1024];
size_t pos = 0;
while (1) {
assert(pos < sizeof(msg));
ssize_t amt = read(client_fd, msg + pos, sizeof(msg) - pos);
if (amt < 0)
err(1, "read failed");
if (amt == 0)
break; // EOF
pos += amt;
char *e = strstr(msg, "\r\n\r\n");
if (e) {
size_t msglen = e - msg;
/* Handle a complete message here */
}
}

这样,当您收到消息 block 时,它们就会写入您的缓冲区。一旦你有了 "\r\n\r\n" 序列,你就可以处理整个消息,即使你可能会以 block 的形式获取它。

关键教训:在 TCP 中,数据包边界和消息边界可能完全不同,并且您从 read 中获得的大小可能仍然不同。您必须通过查看数据本身来找到消息的结尾,而不是通过查看从 read() 返回了多少数据(EOF 除外,它在 read( ) 返回 0)。

脚注:我认为带外数据不会像您认为的那样。

send(client_fd, buffer, sizeof("The msg was: ")+ sizeof(msg), MSG_OOB);

带外数据是 TCP 的一个古怪特性,在现代协议(protocol)中几乎肯定应该避免。它的实现因平台而异,只有发送一个字节的带外数据才是安全的。

Telnet 协议(protocol)(以及基于 Telnet 构建的 FTP)用于某些事情,但在 HTTP 中没有用处。

关于c - 用 C 写一个网络服务器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13103601/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com