作为我之前关于Creating a web server in pure C的帖子的补充,我在发送功能方面遇到了一些问题。这是两个代码片段:
int Send(char *message)
{
int length, bytes_sent;
length = strlen(message);
bytes_sent = send(connecting_socket, message, length, 0);
return bytes_sent;
}
此代码将void *发送到当前套接字。像魅力一样工作!
现在发送SendHTML
void SendHTML(char *Status_code, char *Content_Type, char *HTML)
{
char *head = "\r\nHTTP/1.1 ";
char *content_head = "\r\nContent-Type: ";
char *server_head = "\r\nServer: PT06";
char *length_head = "\r\nContent-Length: ";
char *date_head = "\r\nDate: ";
char *newline = "\r\n";
char Content_Length[100];
int content_length = strlen(HTML);
sprintf(Content_Length, "%i", content_length);
char *message = malloc((
strlen(head) +
strlen(content_head) +
strlen(server_head) +
strlen(length_head) +
strlen(date_head) +
strlen(newline) +
strlen(Status_code) +
strlen(Content_Type) +
strlen(Content_Length) +
content_length +
sizeof(char)) * 2);
if ( message != NULL )
{
time_t rawtime;
time ( &rawtime );
strcpy(message, head);
strcat(message, Status_code);
strcat(message, content_head);
strcat(message, Content_Type);
strcat(message, server_head);
strcat(message, length_head);
strcat(message, Content_Length);
strcat(message, date_head);
strcat(message, (char*)ctime(&rawtime));
strcat(message, newline);
strcat(message, HTML);
Send(message);
free(message);
}
}
如果我要添加
Send("Oh end of HTML Sending eh?");
在Send(message)
之后和free(message)
之前,这不会发送到浏览器?
我认为这可能是HTTP 1.1问题,RFC是否说我只能做一个发送?浏览器收到第一条消息后是否关闭连接?
我如何解决这个问题,以便我可以做到以下几点:
SendHTML("200 OK", "text/plain", "HAaaaii!!");
Send("lolwut?");
这会导致浏览器显示:
HAaaaii !! lolwut?
答案 0 :(得分:6)
由于您发送了内容长度,因此浏览器在“HAaaaii !!”之后将不再接受更多内容因此会解释“lolwut?”作为下一个请求的答案的一部分,这当然会失败。
您可以跳过发送内容长度,但这意味着您没有使用keep-alive,因此无法为每个TCP连接处理多个请求。
每个TCP连接处理多个请求 - 响应交换可以大大加快浏览速度,因为正常的网页由几个单独的资源组成,每个资源都必须单独请求。由于建立TCP连接至少需要3次往返,这会为每个资源增加额外的不必要的延迟。
答案 1 :(得分:4)
在head
变量中,不应有前导"\r\n"
。这可能是导致问题的原因。此外,当你对你的消息进行malloc时,你没有为日期+时间保留足够的空间(根据ctime(3)
的文档,26个字节),所以你几乎肯定会覆盖你的缓冲区并导致堆腐败。
如果您想在知道Content-Length的内容之前开始发送数据,则可以使用chunked transfer encoding。分块传输编码说:“我将继续为你提供大量的数据,我可以告诉你每个块有多长,但我不知道有多少块,直到我完成。 “例如,Google使用分块传输编码。以下是使用cURL生成的Google主页的HTTP请求跟踪:
curl www.google.com --trace-ascii -
(标头已剪切 - 但没有Content-Length标头)
数据流由块大小(十六进制)后跟CRLF,然后是数据字节组成。在这种情况下,只有一个块,但可能会有更多。在最后一个块之后,传输块大小为0,表示最后一个块。
答案 2 :(得分:0)
您是否在每个标题后添加换行符?
您可以使用telnet测试您的应用程序,当然看看输出是什么。
不,不需要只有一个发送。假设流式传输2GB文件。这根本不合逻辑。