已屏蔽 原因:{{ notice.reason }}已屏蔽
{{notice.noticeContent}}
~~空空如也

HTTP比较怪的地方在于,它是非常文本化的协议,不需要写任何结构体,但是非常需要字符串处理技巧。还有,回车符必须用\r\n,而且在两个回车之后还支持二进制文件。

服务器和客户端程序不同,非常看重稳定性,所以不到必须退出的情况下,绝对不要退出。必要的情况下,甚至可以使用try{}catch(...){}和signal(SIG_IGN)处理一下以防意外退出。

Linux下也支持宽字符,只是它们使用UTF-32,并且使用setlocale和wcstombs/mbstowcs进行转换。iconv不建议一般人使用,它基于状态机设计,而不是允许预分配缓冲区的设计,不如setlocale和wcstombs/mbstowcs好用。

  • Linux下wcstombs相当于:对于65001代码页来说,等价于WideCharToMultiByte返回值减1。对于其它代码页来说,等价于WideCharToMultiByte检查最后一个参数pfUsedDefaultChar是否被置位,如果被置位返回-1,否则返回值减1。
  • 以下两个等价
    • setlocale(LC_CTYPE, "xxxxxx");
      retval = wcstombs(dst, src, dstsize);
    • BOOL useddefchr = FALSE;
      retval = WideCharToMultiByte(codepage, 0, src, -1, dst, dstsize, NULL, codepage == 65001 ? NULL : &useddefchr) - 1;
      retval = useddefchr ? -1 : useddefchr;
  • Linux下mbstowcs相当于:等价于MultiByteToWideChar第二个参数置位MB_ERR_INVALID_CHARS,并且返回值减1。
  • 以下两个等价
    • setlocale(LC_CTYPE, "xxxxxx");
      retval = wcstombs(dst, src, dstsize);
    • retval = WideCharToMultiByte(codepage, MB_ERR_INVALID_CHARS, src, -1, dst, dstsize, NULL, &useddefchr) - 1;

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>
文号 / 830613

千古风流
名片发私信
学术分 4
总主题 466 帖总回复 2942 楼拥有证书:进士 学者 笔友
注册于 2009-05-30 21:22最后登录 2019-01-31 17:16
主体类型:个人
所属领域:无
认证方式:邮箱
IP归属地:未同步

个人简介

暂未填写
文件下载
加载中...
{{errorInfo}}
{{downloadWarning}}
你在 {{downloadTime}} 下载过当前文件。
文件名称:{{resource.defaultFile.name}}
下载次数:{{resource.hits}}
上传用户:{{uploader.username}}
所需积分:{{costScores}},{{holdScores}}下载当前附件免费{{description}}
积分不足,去充值
文件已丢失

当前账号的附件下载数量限制如下:
时段 个数
{{f.startingTime}}点 - {{f.endTime}}点 {{f.fileCount}}
视频暂不能访问,请登录试试
仅供内部学术交流或培训使用,请先保存到本地。本内容不代表科创观点,未经原作者同意,请勿转载。
音频暂不能访问,请登录试试
投诉或举报
加载中...
{{tip}}
请选择违规类型:
{{reason.type}}

空空如也

插入资源
全部
图片
视频
音频
附件
全部
未使用
已使用
正在上传
空空如也~
上传中..{{f.progress}}%
处理中..
上传失败,点击重试
等待中...
{{f.name}}
空空如也~
(视频){{r.oname}}
{{selectedResourcesId.indexOf(r.rid) + 1}}
处理中..
处理失败
插入表情
我的表情
共享表情
Emoji
上传
注意事项
最大尺寸100px,超过会被压缩。为保证效果,建议上传前自行处理。
建议上传自己DIY的表情,严禁上传侵权内容。
点击重试等待上传{{s.progress}}%处理中...已上传,正在处理中
空空如也~
处理中...
处理失败
加载中...
草稿箱
加载中...
此处只插入正文,如果要使用草稿中的其余内容,请点击继续创作。
{{fromNow(d.toc)}}
{{getDraftInfo(d)}}
标题:{{d.t}}
内容:{{d.c}}
继续创作
删除插入插入
插入公式
评论控制
加载中...
文号:{{pid}}
加载中...
详情
详情
推送到专栏从专栏移除
设为匿名取消匿名
查看作者
回复
只看作者
加入收藏取消收藏
收藏
取消收藏
折叠回复
置顶取消置顶
评学术分
鼓励
设为精选取消精选
管理提醒
编辑
通过审核
评论控制
退修或删除
历史版本
违规记录
投诉或举报
加入黑名单移除黑名单
查看IP
{{format('YYYY/MM/DD HH:mm:ss', toc)}}
ID: {{user.uid}}