HTTP比较怪的地方在于,它是非常文本化的协议,不需要写任何结构体,但是非常需要字符串处理技巧。还有,回车符必须用\r\n,而且在两个回车之后还支持二进制文件。
服务器和客户端程序不同,非常看重稳定性,所以不到必须退出的情况下,绝对不要退出。必要的情况下,甚至可以使用try{}catch(...){}和signal(SIG_IGN)处理一下以防意外退出。
Linux下也支持宽字符,只是它们使用UTF-32,并且使用setlocale和wcstombs/mbstowcs进行转换。iconv不建议一般人使用,它基于状态机设计,而不是允许预分配缓冲区的设计,不如setlocale和wcstombs/mbstowcs好用。
Linux宽字符版本:
<code class="language-cpp">// winsockhttp1.cpp : 定义控制台应用程序的入口点。 // #include <stdio.h> #include <locale.h> #include <wchar.h> #include <stdlib.h> #include <string.h> #include <netinet in.h> #include <arpa inet.h> #include <unistd.h> int main(int argc, char* argv[]) { // 创建本地 int sock; sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); if (sock == -1) { printf(">> socket error\n"); return 0; } printf(">> socket\n"); // 绑定本地 sockaddr_in addr; addr.sin_addr.s_addr = inet_addr("0.0.0.0"); addr.sin_family = AF_INET; addr.sin_port = htons(8080); int retval = bind(sock, (sockaddr*)&addr, sizeof addr); if (retval == -1) { printf(">> bind error\n"); close(sock); return 0; } printf(">> bind %s:%d\n", inet_ntoa(addr.sin_addr), ntohs(addr.sin_port)); // 监听 retval = listen(sock, 5); if (retval == -1) { printf(">> listen error\n"); close(sock); return 0; } printf(">> listen\n", inet_ntoa(addr.sin_addr), ntohs(addr.sin_port)); while (1) { // 接受连接 sockaddr_in client_addr; unsigned int client_addr_len = sizeof client_addr; int client = accept(sock, (sockaddr*)&client_addr, &client_addr_len); if (client == -1) { printf(">> accept error\n"); continue; } printf(">> accept %s:%d\n", inet_ntoa(client_addr.sin_addr), ntohs(client_addr.sin_port)); // 接收请求内容 char request[65536] = ""; retval = recv(client, request, 65535, 0); if (retval == -1) { printf(">> recv request error\n"); close(client); continue; } printf(">> recv request %d\n", retval); //printf(">> request string:\n%s\n", request); // 纯ASCII内容一般使用28591代码页(ISO-8859-1)处理 setlocale(LC_CTYPE, "C"); char *in_addr = inet_ntoa(client_addr.sin_addr); wchar_t in_addr_wide[32] = L""; mbstowcs(in_addr_wide, in_addr, 32); wchar_t request_wide[65536] = L""; mbstowcs(request_wide, request, 65536); // 通过宽字符生成UTF-8的响应内容 setlocale(LC_CTYPE, ""); wchar_t content_fmt[] = L"<html>" L"<head>" L"<title>测试页面</title>" L"</head>" L"<body>" L"<form action="\"\"" method="\"post\"">" L"<input type="\"text\"" name="\"name\""><br>" L"<textarea name="\"text\""></textarea><br>" L"<input type="\"submit\"">" L"</form>" L"服务器工作正常<br>" L"您的IP地址:%ls<br>" L"您的本地端口号:%d<br>" L"请求头部:" L"<pre>%ls" L"</body>" L"</html>"; wchar_t content_wide[65536] = L""; swprintf(content_wide, 65536, content_fmt, in_addr_wide, ntohs(client_addr.sin_port), request_wide); char content[65536] = ""; wcstombs(content, content_wide, 65536); // 生成响应头部 char header[] = "HTTP/1.1 200 OK\r\n" "Content-Type: text/html; charset=utf-8\r\n" "\r\n"; // 发送响应头部 retval = send(client, header, strlen(header), 0); if (retval == -1) { printf(">> send header error\n"); close(client); continue; } printf(">> send header %d\n", retval); // 发送响应内容 retval = send(client, content, strlen(content), 0); if (retval == -1) { printf(">> send content error\n"); close(client); continue; } printf(">> send content %d\n", retval); // 关闭连接 close(client); printf(">> closesocket\n"); } // 关闭本地 close(sock); printf(">> closesocket\n"); return 0; } </unistd.h></arpa></netinet></string.h></stdlib.h></wchar.h></locale.h></stdio.h>
时段 | 个数 |
---|---|
{{f.startingTime}}点 - {{f.endTime}}点 | {{f.fileCount}} |