在ASP.NET編程中常常遇到下面的代碼片斷,將人員信息以表單方式提交到後臺程序並保存到服務器與數據庫中。html
1 <form action="userManage.ashx" method="post" enctype="application/x-www-form-urlencoded"> 2 名稱: <input type="text" name="uname" class="uname" /><br /> 3 郵件: <input type="text" name="email" class="email" /><br /> 4 <input type="submit" name="submit" value="提交"/> 5 </form>
enctype 屬性規定在發送到服務器以前應該如何對錶單數據進行編碼。數據庫
默認地,表單數據會編碼爲 "application/x-www-form-urlencoded"。就是說,在發送到服務器以前,全部字符都會進行編碼(空格轉換爲 "+" 加號,特殊符號轉換爲 ASCII HEX 值)。編程
enctype屬性值json
值 | 描述 |
---|---|
application/x-www-form-urlencoded | 在發送前編碼全部字符(默認) |
multipart/form-data | 不對字符編碼。瀏覽器 在使用包含文件上傳控件的表單時,必須使用該值。bash |
text/plain | 空格轉換爲 "+" 加號,但不對特殊字符編碼。 |
在《C# HTTP系列8 GET與POST對比說明》中介紹了HTTP的經常使用方法,包含OPTIONS、GET、HEAD、POST、PUT、DELETE、TRACE、CONNECT 這幾種。其中 POST 通常用來向服務端提交數據。更詳細的內容請參考:HTTP1.1協議。服務器
HTTP 協議是以 ASCII 碼 傳輸,創建在 TCP/IP 協議之上的應用層規範。規範把 HTTP 請求分爲三個部分:狀態行、請求頭、消息主體。相似於下面這樣:app
BASH<method> <request-URL> <version> <headers> <entity-body>
協議規定 POST 提交的數據必須放在消息主體(entity-body)中,但協議並無規定數據必須 使用什麼編碼方式。實際上,開發者徹底能夠本身決定消息主體的格式,只要最後發送的 HTTP 請求知足上面的格式就能夠。可是,數據發送出去,還要服務端解析成功纔有意義。通常服務端語言如.NET、JAVA、PHP、Python 等,以及它們的 framework,都內置了自動解析常見數據格式的功能。函數
服務端一般是根據請求頭(headers)中的 Content-Type 字段來獲知請求中的消息主體是用何種方式編碼,再對主體進行解析。
因此說到 POST 提交數據方案,包含了 Content-Type 和消息主體編碼方式兩部分。post
enctype
屬性,那麼最終就會以 application/x-www-form-urlencoded 方式提交數據。
1 <form id="form1" runat="server" action="UserManageHandler.ashx" method="post" enctype="application/x-www-form-urlencoded"> 2 <div> 3 名稱: <input type="text" name="uname" class="uname" /><br /> 4 郵件: <input type="text" name="email" class="email" /><br /> 5 <input type="submit" name="submit" value="提交" /> 6 </div> 7 </form>
此點擊【提交】按鈕,Form提交請求數據,Fiddler抓包時看到的請求以下(無關的請求頭在本文中都省略掉了):
首先,Content-Type 被指定爲 application/x-www-form-urlencoded;
其次,提交的數據按照 key1=val1&key2=val2 的方式進行編碼,key 和 val 都進行了 URL 轉碼。
大部分服務端語言都對這種方式有很好的支持。例如 .NET 中,context.Request["uname"]能夠獲取到名稱的值,context.Request["email"]能夠獲得郵件的值。
不少時候,用 Ajax 提交數據時,也是使用這種方式。
例如 JQuery(谷歌公司) 和 QWrap(百度公司) 的 Ajax,Content-Type 默認值都是「application/x-www-form-urlencoded;charset=utf-8」。
<form id="form1" runat="server" action="UserManageHandler.ashx" method="post" enctype="multipart/form-data"> <div> 名稱: <input type="text" name="uname" class="uname" /><br/> 郵件: <input type="text" name="email" class="email" /><p/> 附件1: <input type="file" name="file1" class="file" /><p/> 附件2: <input type="file" name="file2" class="file" /><p/> 附件3: <input type="file" name="file3" class="file" /><p/> <input type="submit" name="submit" value="提交" /> </div> </form>
- application/x-www-form-urlencoded 不能用於上傳文件,只能提交文本,固然若是有file控件的話也只能提交文件名。
- multipart/form-data 用於上傳文件以及文本。
此點擊【提交】按鈕,Form提交請求數據,Fiddler抓包時看到的請求以下(無關的請求頭在本文中都省略掉了):
方式二:上傳多個附件,一個普通文本,一個Office word文檔,一個png圖片
此點擊【提交】按鈕,Form提交請求數據,Fiddler抓包時看到的請求以下(無關的請求頭在本文中都省略掉了):
(1)boundary:用於分割不一樣的字段,爲了不與正文內容重複。以2個橫線「--」開頭,最後的字段以後以2個橫線「--」結束。
(2)Content-Type: 指明瞭數據是以 multipart/form-data 來編碼。
(3)消息主體裏按照字段個數又分爲多個結構相似的部分,每部分都是以 --boundary
開始,緊接着是內容描述信息,而後是回車,最後是字段具體內容(文本或二進制)。若是傳輸的是文件,還要包含文件名和文件類型信息。消息主體最後以 --boundary--
標示結束。
關於 multipart/form-data 的詳細定義,請查看 rfc1867
這種方式通常用來上傳文件,各大服務端語言對它也有着良好的支持。
上面提到的這兩種 POST 數據的方式,都是瀏覽器原生支持的,並且現階段標準中原生 <form> 表單也只支持這兩種方式(經過 <form> 元素的 enctype
屬性指定,默認爲 application/x-www-form-urlencoded
。)。
隨着愈來愈多的 Web 站點,尤爲是 WebApp,所有使用 Ajax 進行數據交互以後,咱們徹底能夠定義新的數據提交方式,給開發帶來更多便利。
enctype
還支持 text/plain
,不過用得很是少。
<form action="userManage.ashx" method="post" enctype="text/plain"> 名稱: <input type="text" name="uname" class="uname" /><br /> 郵件: <input type="text" name="email" class="email" /><br /> <input type="submit" name="submit" value="提交"/> </form>
此點擊【提交】按鈕,Form提交請求數據,Fiddler抓包時看到的請求以下(無關的請求頭在本文中都省略掉了):
application/json
application/json 這個 Content-Type 做爲響應頭你們確定不陌生。實際上,如今愈來愈多的人把它做爲請求頭,用來告訴服務端消息主體是序列化後的 JSON 字符串。因爲 JSON 規範的流行,除了低版本 IE 以外的各大瀏覽器都原生支持 JSON.stringify,服務端語言也都有處理 JSON 的函數,使用 JSON 不會遇到什麼問題。
一、form-data:
http請求中的multipart/form-data,它會將表單的數據處理爲一條消息,以標籤爲單元,用分隔符分開。既能夠上傳鍵值對,也能夠上傳文件。當上傳的字段是文件時,會有Content-Type來表名文件類型;content-disposition,用來講明字段的一些信息;
點擊【Code】按鈕,打開以下窗體
二、x-www-form-urlencoded:
就是application/x-www-from-urlencoded,會將表單內的數據轉換爲鍵值對
點擊【Code】按鈕,打開以下窗體
三、raw
能夠上傳任意格式的文本,能夠上傳text、json、xml、html等
後臺代碼以下:
1 public void ProcessRequest(HttpContext context) 2 { 3 context.Response.ContentType = "application/json"; //"text/plain"; 4 5 string uname = context.Request["uname"]; 6 string email = context.Request["email"]; 7 8 StringBuilder sbFiles = new StringBuilder(); 9 HttpFileCollection filesCollection = context.Request.Files; 10 11 if (filesCollection != null && filesCollection.Count > 0) 12 { 13 for (var i = 0; i < filesCollection.Count; i++) 14 { 15 HttpPostedFile postedFile = filesCollection[i]; 16 if (!string.IsNullOrWhiteSpace(postedFile.FileName)) 17 { 18 sbFiles.AppendLine(); 19 sbFiles.AppendLine("附件" + (i + 1)); 20 sbFiles.AppendLine("文件名稱:" + postedFile.FileName); 21 sbFiles.AppendLine("文件大小(字節):" + postedFile.ContentLength); 22 sbFiles.AppendLine("客戶端發送的文件的 MIME 內容類型:" + postedFile.ContentType); 23 } 24 } 25 } 26 27 context.Response.Write("提交結果以下:" + Environment.NewLine + 28 "名稱:" + uname + Environment.NewLine + 29 "郵箱:" + email + Environment.NewLine + 30 sbFiles 31 ); 32 }
點擊【提交】按鈕,返回以下結果:
四、binary
至關於Content-Type:application/octet-stream,只能夠上傳二進制數據。
一般用來上傳文件,因爲沒有鍵值,因此,一次只能上傳一個文件。