Http服务器实现文件上传与下载(四)(1)


一、引言

欢迎大家来到和我一起编写Http服务器实现文件的上传和下载,现在我稍微回顾一下之前我说的,第一、二章说明说明了整体的HTTP走向,第三章实现底层的网络编程。接着这一章我想给大家讲的是请求获取,和响应发送的内容。这里主要讲解的响应内容,为什么?因为我们编写的是一个与浏览器交互的HTTP服务器,所以大多数的情况下我们只进行被动的应答。

这就是一种"提问--回答"的问题。其实在讲解这章的时候,我本来准备给大家讲解一下Linux一些信号中断的问题。因为在网络层发送的时候,系统会发送一些信号给我们的应用程序,所以会导致我们的程序意外的终止。但当我写的这篇博客的时候我又放弃,我想在讲流程走向的时候再提一个中断捕获吧。在这个请求响应层的类其实真正的设计需要很多的内容,这里就是HttpResponse类和HttpRequest类的设计,在j2EE中,我们编写Servlet的时候就用到了这2个类,如HttpServletResquest,HttpServletResponse的类,如果对这里面的内容感兴趣,可以下载tomcat,在servlet-api.jar包里面有这些类。

在本文的实现中,Request类只包含了一个获取请求头和解析头的一些方法。如何解析头,我在《Http服务器实现文件上传与下载(一)》已经讲解了,读者只需要对其封装一个类即可。

二、HttpRequest类

请求消息的解析是通过被定义在命名空间为Http的类名为HttpRequest。这个类的构造函数接受一个套接字,就是跟我们连接的那个套接字,在网络层我们已经讲过了,然后在getHeader方法中调用server_read()获取请求头,然后通过Utils::parseHeader()函数进行解析。这样把解析的内容放入需要的string中,当前不太需要的直接在map里面。这里我直接贴出代码,大家看起来也比较容易。这里我在这一章节我主要讲解的是文件的下载,所以主要会对HttpResponse的类的分析,而HttpRequest类只贴出目前需要的内容。

头文件(include/httprequest.h)

1 #ifndef HTTPREQUEST_H
2 #define HTTPREQUEST_H
3 #include "socket.h"
4 #include 
5 #include 
6 #include 
7 namespace Http{
8 class HttpRequest{
9 public:
10 HttpRequest(TCP::Socket &c);
11 virtual ~HttpRequest();
12 std::map
13 ......
14 protected:
15 private:
16 std::string method;
17 std::string url;
18 std::string host;
19 TCP::Socket &s;
20 };
21 }
22
23 #endif // HTTPREQUEST_H

源文件(src/httprequest.cpp)

1 #include "httprequest.h"
2 #include "utils.h"
3 namespace Http{
4 HttpRequest::HttpRequest(TCP::Socket &c):s(c){
5 }
6
7 HttpRequest::~HttpRequest(){
8 }
9 std::map
10 char recvBuf[1024];
11 memset(recvBuf,0,sizeof(recvBuf));
12 s.server_read(confd,recvBuf,1024);
13 std::cout<
14 std::map
15 method =mp["Method"];
16 url=mp["Url"];
17 host=mp["Host"];
18 return mp;
19 }
20 ......
21 }

三、HttpResponse类

当我们访问Http服务器的时候,浏览器显示可以下载的文件的内容,然后我们点击需要下载的文件,然后文件就可下载了。首先我点击这个文件这个URL时,浏览器给我们发送一些请求头,例如它发送一个为/download/HttpServer.zip这个URL,说明他需要下载的文件,而且该文件为HttpServer.zip。在上面我们已经可以用getHeader来捕获这个请求头,然后获取这个URL。之后服务端还是要发送一个响应头,告诉浏览器你的请求我们同意,请求头结束以空行为标记,接着就是具体的文件的内容了。

在发送响应头时,还是需要发送协议版本,状态码,响应内容类型,文件的长度,文件断点下载等内容,或者传输的时候采用chunk传输,但是这里我采用文件的长度来标记。读者可以自行查看其它方式传输内容。特别要注意ed是一定要在响应头中指定传输实体的大小,否则客户端不知道什么时候结束,这时可能拒绝接收服务端发来的字节。在这个类中,请求下载的文件发送时,我采用sendFile这个函数,这个函数读取文件就是采用二进制的方式,并且在响应头中也告知浏览器以二进制的方式接收文件。这样都是以二进制的方式读取和发送文件才不会出现问题。sendLineFile 和sendIndexFile两者大致相同,都是采用ASCII文本的方式发送内容,这样比如HTML这些需要显示在浏览器的内容,可以通过这两个函数。通过函数名可知在sendLineFile会以文件行的方式读取,而sendIndexFile文件会把内容写在同一行上。例如:我们浏览器请求一个index.html的内容,这时采用2个sendLineFile和sendIndexFile的显示效果都是一样的,但是如果点击右键查看源码时,sendLineFile的内容是以源文件一样的,而sendIndexFile发送的内容会都在第一行,不会换行。

说了这么多大家也比较清楚了,下面贴出具体一些代码。

头文件(include/httpresponse.h)

1 #ifndef HTTPRESPONSE_H
2 #define HTTPRESPONSE_H
3 #include "socket.h"
4 #include
5 #include
6 #include
7 #include
8 #include
9 #include
10 #include "utils.h"
11 namespace Http{
12 class HttpResponse{
13 public:
14 HttpResponse(TCP::Socket &c);
15 virtual ~HttpResponse();
16 ssize_t send(int confd,std::string content);
17 ssize_t sendIndexFile(int confd,std::string FileName);
18 ssize_t sendFile(int &confd,std::string FileName,int64_t pos);
19 ssize_t sendLineFile(int confd,std::string file);
20 void setProtocal(std::string);
21 void setStatusCode(std::string);
22 void setServerName(std::string);
23 void setContentType(std::string);
24 void setContentRange(std::string);
25 void setContentLength(int64_t);
26 protected:
27 std::string getHeader() const;
28 private:
29 std::string protocal;
30 std::string statusCode;
31 std::string serverName;
32 std::string contentType;
33 std::string contentLength;
34 std::string contentRange;
35 std::string connection;
36 std::string date;
37 TCP::Socket &s;
38 };
39 }
40 #endif // HTTPRESPONSE_H




相关内容