Http服務(wù)器實(shí)現(xiàn)文件上傳與下載(五)
一、引言
歡迎大家和我一起編寫Http服務(wù)器實(shí)現(xiàn)文件的上傳和下載,現(xiàn)在我回顧一下在上一章節(jié)中提到的一些內(nèi)容,之前我已經(jīng)提到過文件的下載,在文件的下載中也提到了文件的續(xù)下載只需要在響應(yīng)頭中填寫Content-Range這一字段,并且服務(wù)器的文件指針指向讀取的指定位置開始讀取傳輸。在這一章節(jié)中我講講解文件的上傳這一功能,講完這一章節(jié),大致的功能也全部完成,接著就是上面文件控制模塊和一些資源模塊。
在文件的上傳中主要以HttpRequest類為主,在考慮文件的上傳時(shí)我一點(diǎn)迷惑,到底把文件的上傳功能是放到HttpResponse下還是在HttpRequest下,畢竟HttpResponse中有一些相應(yīng)的文件下載功能,在添加一個(gè)文件上傳功能也不為過。但是我最終還是選擇在HttpRequest中,原因是我主要是HttpResponse作為是服務(wù)器到瀏覽器發(fā)送內(nèi)容,而HttpRequest作為瀏覽器到服務(wù)器發(fā)送內(nèi)容。這樣下載和上傳的功能就分別坐落在了HttpResponse和HttpRequest上了。
在完成功能上的歸屬問題后,接著直接上代碼,在文件的上傳中,涉及到C++流。在這里其實(shí)用到不是很多的內(nèi)容,但是這卻是C++一個(gè)重要的一大塊內(nèi)容。有時(shí)間和大家在一起復(fù)習(xí)這一塊內(nèi)容。好了,接著上代碼咯,上一章的內(nèi)容有設(shè)計(jì)一些HttpRequest的代碼,沒有全部的包括進(jìn)去。
二、HttpRequest
頭文件(include/httprequest.h)
1 #ifndef HTTPREQUEST_H 2 #define HTTPREQUEST_H 3 #include "socket.h" 4 #include 5 #include6 #include 7 namespace Http{ 8 class HttpRequest{ 9 public: 10 HttpRequest(TCP::Socket &c); 11 virtual ~HttpRequest(); 12 std::string getMethod() const; 13 std::string getUrl() const; 14 std::string getHost() const; 15 std::map 16 ssize_t upload(int confd,std::string filename); 17 protected: 18 private: 19 std::string method; 20 std::string url; 21 std::string host; 22 TCP::Socket &s; 23 }; 24 } 25 #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 ssize_t HttpRequest::upload(int confd,std::string filename){
21 char buf[1024];
22 size_t n=0;
23 ssize_t nread=0;
24 std::string boundary;
25 std::string file;
26 std::ofstream outStream;
27 int readlineCount=1;
28 while(1){
29 memset(buf,0,sizeof(buf));
30 n=s.server_readline(confd,buf,sizeof(buf));
31 if(readlineCount==1){
32 boundary=std::string(buf,buf+strlen(buf)-2);
33 boundary+="--\r\n";
34 std::cout<
#p#
35 }else if(readlineCount==2){
36 int i=n;
37 while(buf[i]!='='){
38 if((buf[i]>='0'&&buf[i]<='9')
39 ||(buf[i]>='a'&&buf[i]<='z')
40 ||(buf[i]>='A'&&buf[i]<='Z')
41 ||(buf[i]=='.'))
42 i--;
43 else{
44 buf[i]='*';
45 i--;
46 }
47 }
48 file=std::string(buf+i+2,buf+n-3);
49 }else if(readlineCount==3){
50 std::string rw;
51 rw=std::string(buf,buf+strlen(buf));
52 int pos=rw.find('/');
53 rw=rw.substr(0,pos);
54 filename=filename+file;
55 if(rw=="Content-Type: text")
56 outStream.open(filename.c_str());
57 else{
58 outStream.open(filename.c_str(),std::ios::binary);
59 std::cout<<"ios::binary"<
60 }
61 }else if(readlineCount==4){
62 memset(buf,0,sizeof(buf));
63 while(1){
64 n=s.server_readn(confd,buf,sizeof(buf));
65 if(n==boundary.size()&&strcmp(buf,boundary.c_str())==0){
66 goto exit;
67 }
68 nread+=n;
69 if(buf[n-1]==0){
70 outStream.write(buf,n-1);
71 }else{
72 outStream.write(buf,n);
73 }
74 }
75 }
76 readlineCount++;
77 }
78 exit:
79 outStream.close();
80 s.server_close(confd);
81 return nread;
82 }
83 std::string HttpRequest::getMethod() const{
84 return method;
85 }
86 std::string HttpRequest::getUrl() const{
87 return url;
88 }
89 std::string HttpRequest::getHost() const{
90 return host;
91 }
92 }
好了上傳文件的代碼也已經(jīng)出來了,現(xiàn)在就是對(duì)其稍微的解釋一下把。在解釋代碼之前先看一下我們?cè)邳c(diǎn)擊上傳文件按鈕的時(shí)候,瀏覽器給服務(wù)器發(fā)送的內(nèi)容是什么,比如我有一個(gè)test.txt的文本文件(這里采用文件文件是為了好查看內(nèi)容,其實(shí)二進(jìn)制文件也是一致的)。test.txt文件的內(nèi)容只有一行就是aaabbb這6個(gè)字母。接著打開可以火狐的開發(fā)者網(wǎng)絡(luò)這一功能。并且點(diǎn)擊發(fā)送文件后,可以在消息頭上看到如下信息。

這些內(nèi)容在之前的章節(jié)已經(jīng)講過了,這里就不重復(fù)了,并且點(diǎn)擊參數(shù)這一選項(xiàng)可以看到如下信息。
 
在這里第1,2行是請(qǐng)求頭的內(nèi)容,接著一行空行之后是請(qǐng)求體4-9行。看到請(qǐng)求體的內(nèi)容不是直接是test.txt的內(nèi)容。顯示‘--23469111452’為開頭,拜師這個(gè)是文本的分隔符。前面固定一段'-',加上一個(gè)瀏覽器自動(dòng)產(chǎn)生的數(shù)據(jù)。并且一個(gè)文件的解釋也是這樣,只是數(shù)字后面多了2個(gè)'-'。在第5,6行是對(duì)上傳的文件的描述。接著是一行空行。第8行開始就是文件的內(nèi)容了。知道這個(gè)請(qǐng)求體后,很容易的就可以寫出代碼。上面的的upload中readlineCount變量就是起到定位功能??捶?wù)器已經(jīng)接收到那一行了,這里s.server_readn這個(gè)行數(shù)之間沒有提交,現(xiàn)在的代碼段一直在修改,所以有些與博客有點(diǎn)差別,大體上還是一致的。















 
 
 
 
 
 
 