文章目錄
在C#中有HttpWebRequest類,可以很方便用來擷取http請求,但是這個類對Post方式沒有提供一個很方便的方法來擷取資料。網上有很多人提供瞭解決方法,但都參差不齊,這裡我把我使用的方法總結出來,與大家分享。
本文精華:實現了post的時候即可以有字串的key-value,還可以帶檔案。
Post資料格式
Post提交資料的時候最重要就是把Key-Value的資料放到http請求流中,而HttpWebRequest沒有提供一個屬性之類的東西可以讓我們自由添加Key-Value,因此就必須手工構造這個資料。
根據RFC 2045協議,一個Http Post的資料格式如下:
Content-Type: multipart/form-data; boundary=AaB03x --AaB03x Content-Disposition: form-data; name="submit-name" Larry --AaB03x Content-Disposition: form-data; name="file"; filename="file1.dat" Content-Type: application/octet-stream ... contents of file1.dat ... --AaB03x--
詳細解釋如下:
Content-Type: multipart/form-data; boundary=AaB03x
如上面所示,首先聲明資料類型為multipart/form-data, 然後定義邊界字串AaB03x,這個邊界字串就是用來在下面來區分各個資料的,可以隨便定義,但是最好是用破折號等資料中一般不會出現的字元。然後是分行符號。
注意:Post中定義的分行符號是\r\n
--AaB03x
這個是邊界字串,注意每一個邊界符前面都需要加2個連字號“--”,然後跟上分行符號。
Content-Disposition: form-data; name="submit-name"
這裡是Key-Value資料中字串類型的資料。 submit-name 是這個Key-Value資料中的Key。當然也需要分行符號。
Larry
這個就是剛才Key-Value資料中的value。
--AaB03x
邊界符,表示資料結束。
Content-Disposition: form-data; name="file"; filename="file1.dat"
這個代表另外一個資料,它的key是file,檔案名稱是file1.dat。 注意:最後面沒有分號了。
Content-Type: application/octet-stream
這個標識檔案類型。application/octet-stream表示位元據。
... contents of file1.dat ...
這個是檔案內容。可以使二進位的資料。
--AaB03x--
資料結束後的分界符,注意因為這個後面沒有資料了所以需要在後面追加一個“--”表示結束。
C#下Post資料的函數
搞明白格式後,我們就很容易寫出C#的代碼了。如下所示:
private static string HttpPostData(string url, int timeOut, string fileKeyName, string filePath, NameValueCollection stringDict){ string responseContent; var memStream = new MemoryStream(); var webRequest = (HttpWebRequest)WebRequest.Create(url); // 邊界符 var boundary = "---------------" + DateTime.Now.Ticks.ToString("x"); // 邊界符 var beginBoundary = Encoding.ASCII.GetBytes("--" + boundary + "\r\n"); var fileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read); // 最後的結束符 var endBoundary = Encoding.ASCII.GetBytes("--" + boundary + "--\r\n"); // 設定屬性 webRequest.Method = "POST"; webRequest.Timeout = timeOut; webRequest.ContentType = "multipart/form-data; boundary=" + boundary; // 寫入檔案 const string filePartHeader = "Content-Disposition: form-data; name=\"{0}\"; filename=\"{1}\"\r\n"+ "Content-Type: application/octet-stream\r\n\r\n"; var header = string.Format(filePartHeader, fileKeyName, filePath); var headerbytes = Encoding.UTF8.GetBytes(header); memStream.Write(beginBoundary, 0, beginBoundary.Length); memStream.Write(headerbytes, 0, headerbytes.Length); var buffer = new byte[1024]; int bytesRead; // =0 while ((bytesRead = fileStream.Read(buffer, 0, buffer.Length)) != 0) { memStream.Write(buffer, 0, bytesRead); } // 寫入字串的Key var stringKeyHeader = "\r\n--" + boundary + "\r\nContent-Disposition: form-data; name=\"{0}\""+ "\r\n\r\n{1}\r\n"; foreach (byte[] formitembytes in from string key in stringDict.Keys select string.Format(stringKeyHeader, key, stringDict[key]) into formitem select Encoding.UTF8.GetBytes(formitem)) { memStream.Write(formitembytes, 0, formitembytes.Length); } // 寫入最後的結束邊界符 memStream.Write(endBoundary, 0, endBoundary.Length); webRequest.ContentLength = memStream.Length; var requestStream = webRequest.GetRequestStream(); memStream.Position = 0; var tempBuffer = new byte[memStream.Length]; memStream.Read(tempBuffer, 0, tempBuffer.Length); memStream.Close(); requestStream.Write(tempBuffer, 0, tempBuffer.Length); requestStream.Close(); var httpWebResponse = (HttpWebResponse)webRequest.GetResponse(); using (var httpStreamReader = new StreamReader(httpWebResponse.GetResponseStream(), Encoding.GetEncoding("utf-8"))) { responseContent = httpStreamReader.ReadToEnd(); } fileStream.Close(); httpWebResponse.Close(); webRequest.Abort(); return responseContent;}
參考資料
- Rfc2045:http://www.w3.org/TR/html401/interact/forms.html#h-17.13.4.
- http://stackoverflow.com/questions/566462/upload-files-with-httpwebrequest-multipart-form-data