c语言http过滤字符,C语言编写Http服务器中Request的解析,让一些从零开始

制作流程

观察收到的http数据

解析 request 的 method url version

解析 header

解析 body

观察收到的http数据

69d62f6cb4142316e1259d027e7046ec.png

如果你能完成一个简单的基于TCP/IP的socket server 程序,那么恭喜你,本文你可以好好读一读。HTTP正式基于TCP/IP的应用层协议,所以只要我们的程序能读懂HTTP数据,并做出符合HTTP协议的响应,那么就能完成HTTP的通信。

如果你有机会,可以常识使用telnet连接我们的服务器,你将的得到得是一些没有意义的字符。如果是浏览器,会传送什么呢?我们试着在浏览器地址栏输入我们的 服务器 地址: 127.0.0.1:9734 后访问,发现浏览器说“127.0.0.1 发送的响应无效。”, 那是说我们返回给浏览器的数据浏览器读不懂,因为现代的浏览器默认用http协议请求访问我们的服务器,而我们的返回的数据只是"helloworld"字符串,并不符合http协议的返回格式。虽然如此,但浏览器却是很有诚意的给我们的服务器发标准的http请求,不信我们看下我们的服务器收到的信息:

64394995c817fd2ab3d726cc88211d3a.png

先观察一会儿,看起来第一行是http请求的类型,第二行开始是一些":"号分割的键值对。的确如此,第一行告诉我们是用的GET请求,请求的url是"/",用的是1.1的HTTP版本。第二行开始是HTTP的请求头部。除了GET请求外,另一种常用的请求是POST。用浏览器发POST请求稍麻烦,我们就借用curl工具来发送个HTTP POST请求给服务器看下数据又会是怎们样的:curl -d "message=nice to meet you" 127.0.0.1:9734/hello

服务器收到的信息:

710498009d36370ca8e6e8d707bf501e.png

可以看到头部信息之后多了一空行和之后的POST的body数据信息。还要注意的是Content-Length头,代表POST的body数据的大小。

解析 request 的 method url version

先来解析最简单的第一行: "POST /hell HTTP/1.1", 只需要用空格split出三个字符串就好了。

1.request.h

84c81d82cb78f2c45d7421ae8f3d01d6.png

2.request.c

442ad6af76687b778f94f9f1a2f720d4.png

3.编写测试用例

3393bdec0d5e3aa25a3f3fedab8f5d35.png

在test目录下执行:` gcc ../request.h ../request.c requestTest.c && ./a.out`,可以看到我们解析的方法正确。

解析 header

header的解析看起来比较复杂,每一行很容易看出是用":"分割的key-value对,所以我们可以用HashMap来表达。如何判断header数据的结束呢,通过前面的观察,可以发现如果是POST会有一个空行和body隔开,是GET的话只能检查客户端的数据是否发完,发完就代表header也结尾了。在正式解析header之前,我们先构造基本数据的数据结构,以方便以后使用。

1. 创建链表结构体

2. 创建哈希表结构体

3. 按行解析header,遇到空行或字符串结尾停止

因为代码比较多,这里不久不把两种结构体得实现代码贴出来,如果足下需要可以私信我"代码"获取。

解析header代码,有了哈希结构体后,解析header就方便多了,只要按行根据":" 拆分成 key和value就行了

关键代码:

ffc072fcdfca437186a6bad48d57c8c3.png

解析body

解析body很简单,如果最后一行不是空格不是空行,说明是有body数据的,空行后面的就是body数据了.

header里面有个关键的key, ‘Content-Length’ 代表了body有多长,我们可以利用这个字段来判断body的结尾。

13b5a3087b0dee39d8fb025061ae1a79.png

大功告成 最后打印我们的成果

打印解析内容

b075ef3655346a9704ea68cb84550f25.png

执行 gcc request.h request.c main.c tools/utils.c tools/utils.h && ./a.out

然后新开一个终端执行 curl -d "message=nice to meet you" 127.0.0.1:9734/hello-everyone

看到输出结果:

83582a531140ec720e661b744b00343d.png

总的来说难度系数还是有一些得呦,再见。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值