在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 02. 03. --AaB03x 04. Content-Disposition: form-data; name="submit-name" 05. 06. Larry 07. --AaB03x 08. Content-Disposition: form-data; name="file"; filename="file1.dat" 09. Content-Type: application/octet-stream 10. 11. ... contents of file1.dat ... 12. --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, 02. string filePath, NameValueCollection stringDict) 03.{ 04. string responseContent; 05. var memStream = new MemoryStream(); 06. var webRequest = (HttpWebRequest)WebRequest.Create(url); 07. // 边界符 08. var boundary = "---------------" + DateTime.Now.Ticks.ToString("x"); 09. // 边界符 10. var beginBoundary = Encoding.ASCII.GetBytes("--" + boundary + "\r\n"); 11. var fileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read); 12. // 最后的结束符 13. var endBoundary = Encoding.ASCII.GetBytes("--" + boundary + "--\r\n"); 14. 15. // 设置属性 16. webRequest.Method = "POST"; 17. webRequest.Timeout = timeOut; 18. webRequest.ContentType = "multipart/form-data; boundary=" + boundary; 19. 20. // 写入文件 21. const string filePartHeader = 22. "Content-Disposition: form-data; name=\"{0}\"; filename=\"{1}\"\r\n"+ 23. "Content-Type: application/octet-stream\r\n\r\n"; 24. var header = string.Format(filePartHeader, fileKeyName, filePath); 25. var headerbytes = Encoding.UTF8.GetBytes(header); 26. 27. memStream.Write(beginBoundary, 0, beginBoundary.Length); 28. memStream.Write(headerbytes, 0, headerbytes.Length); 29. 30. var buffer = new byte[1024]; 31. int bytesRead; // =0 32. 33. while ((bytesRead = fileStream.Read(buffer, 0, buffer.Length)) != 0) 34. { 35. memStream.Write(buffer, 0, bytesRead); 36. } 37. 38. // 写入字符串的Key 39. var stringKeyHeader = "\r\n--" + boundary + 40. "\r\nContent-Disposition: form-data; name=\"{0}\""+ 41. "\r\n\r\n{1}\r\n"; 42. 43. foreach (byte[] formitembytes in from string key in stringDict.Keys 44. select string.Format(stringKeyHeader, key, stringDict[key]) 45. into formitem 46. select Encoding.UTF8.GetBytes(formitem)) 47. { 48. memStream.Write(formitembytes, 0, formitembytes.Length); 49. } 50. 51. // 写入最后的结束边界符 52. memStream.Write(endBoundary, 0, endBoundary.Length); 53. 54. webRequest.ContentLength = memStream.Length; 55. 56. var requestStream = webRequest.GetRequestStream(); 57. 58. memStream.Position = 0; 59. var tempBuffer = new byte[memStream.Length]; 60. memStream.Read(tempBuffer, 0, tempBuffer.Length); 61. memStream.Close(); 62. 63. requestStream.Write(tempBuffer, 0, tempBuffer.Length); 64. requestStream.Close(); 65. 66. var httpWebResponse = (HttpWebResponse)webRequest.GetResponse(); 67. 68. using (var httpStreamReader = new StreamReader(httpWebResponse.GetResponseStream(), 69. Encoding.GetEncoding("utf-8"))) 70. { 71. responseContent = httpStreamReader.ReadToEnd(); 72. } 73. 74. fileStream.Close(); 75. httpWebResponse.Close(); 76. webRequest.Abort(); 77. 78. return responseContent; 79.}