HttpUtility.UrlEncode 在 Encode 的時候, 將空格轉換成加號('+'), 在 Decode 的時候將加號轉爲空格, 可是瀏覽器是不能理解加號爲空格的, 因此若是文件名包含了空格, 在瀏覽器下載獲得的文件, 空格就變成了加號 在用 ASP.Net 開發頁面的時候, 咱們經常經過 System.Web.HttpUtility.UrlEncode 和 UrlDecode 在頁面間經過 URL 傳遞參數. 成對的使用 Encode 和 Decode 是沒有問題的. 可是, 咱們在編寫文件下載的頁面的時候, 經常用以下方法來指定下載的文件的名稱: Response.AddHeader("Content-Disposition","attachment; filename=" + HttpUtility.UrlEncode(fileName, Encoding.UTF8)); 之因此轉換成 UTF8 是爲了支持中文文件名. 這時候問題就來了, 由於 HttpUtility.UrlEncode 在 Encode 的時候, 將空格轉換成加號('+'), 在 Decode 的時候將加號轉爲空格, 可是瀏覽器是不能理解加號爲空格的, 因此若是文件名包含了空格, 在瀏覽器下載獲得的文件, 空格就變成了加號. 一個解決辦法是, 在 HttpUtility 的 UrlEncode 以後, 將 "+" 替換成 "%20"( 若是原來是 "+" 則被轉換成 "%2b" ) , 如: fileName = HttpUtility.UrlEncode(fileName, Encoding.UTF8); fileName = fileName.Replace("+", "%20"); 不明白微軟爲何要把空格轉換成加號而不是"%20". 記得 JDK 的 UrlEncoder 是將空格轉換成 "%20"的. 經檢查, 在 .Net 2.0 也是這樣. HttpUtility.UrlEncode 在 Encode 的時候, 將空格轉換成加號('+'), 在 Decode 的時候將加號轉爲空格, 可是瀏覽器是不能理解加號爲空格的, 因此若是文件名包含了空格, 在瀏覽器下載獲得的文件, 空格就變成了加號 <html> <head> <meta http-equiv='Content-Type' content='text/html; charset=gb2312'> <title>URL解碼(Decode)/編碼(Encode)</title> </head> <body> <center><font color=green size=+2>URL解碼(Decode)/編碼(Encode)</font><br> 須要解碼的字符串:<TEXTAREA ID="String1" ROWS="10" COLS="30"></TEXTAREA> 解碼後的字符串:<TEXTAREA ID="String2" ROWS="10" COLS="30"></TEXTAREA><br> 須要編碼的字符串:<TEXTAREA ID="String3" ROWS="10" COLS="30"></TEXTAREA> 編碼後的字符串:<TEXTAREA ID="String4" ROWS="10" COLS="30"></TEXTAREA><br> <INPUT TYPE="button" ID="Decode" value="解碼(Decode)" onClick="javascript:String2.value=decodeURI(String1.value);"> <INPUT TYPE="button" ID="Encode" value="編碼(Encode)" onClick="javascript:String4.value=encodeURI(String3.value);"> </center> </body> </html> url傳遞中文的解決方案總結 1.設置web.config文件。(我不喜歡設置成這樣) <system.web> ...... <globalization requestEncoding="gb2312" responseEncoding="gb2312" culture="zh-CN" fileEncoding="gb2312" /> ...... </system.web> 或者: aspx文件中: <meta http-equiv="Content-Type" content="text/html; charset=gb2312"> 2.傳遞中文以前,將要傳遞的中文參數進行編碼,在接收時再進行解碼。 >> 進行傳遞 string Name = "中文參數"; Response.Redirect("B.aspx?Name="+Server.UrlEncode(Name)); >> 進行接收 string Name = Request.QueryString["Name"]; Response.Write(Server.UrlDecode(Name)); 或者: NavigateURL='<%# "WebForm2.aspx?Singer=" + HttpUtility.UrlEncode("中國人", System.Text.Encoding.GetEncoding("GB2312")) %>' 3.若是是從 .HTML 文件向 .Aspx 文件進行傳遞中文參數的話(即不從後臺用 Redirect()方法進行 Url 轉換)。同樣要將傳遞的中文參數進行編碼,在接收時再進行解碼。 >> 進行傳遞 <script language="JavaScript"> function GoUrl() { var Name = "中文參數"; location.href = "B.aspx?Name="+escape(Name); } </script> <body onclick="GoUrl()"> >> 進行接收 string Name = Request.QueryString["Name"]; Response.Write(Server.UrlDecode(Name)); 通常來講。設置web.config文件就能夠了。可是若是你用 JavaScript 調用 webservice 方法的話(往webservice裏面傳遞中文參數)。設置 web.config 文件好象無效。 ———————————————————— 在html中實現編解碼: <script language="javascript"> function openUrl(src) { var strUrl=escape(src); window.open(strUrl); } function change_url(src) { document.location.href=escape(src); } </script> 在新窗口保存 <a href='javascript:openUrl("css/20040603123628交易中心菸葉網上集中交易系統合同.doc");' >20040603123628交易中心網上集中交易系統合同</a> 當前位置保存,無閃爍。 <a href="#" onclick=javascript:change_url("css/20040603123628交易中心菸葉網上集中交易系統合同.doc")>20040603123628交易中心網上集中交易系統合同</a> 注意:路徑中的斜線是:「/」,而不是「/」,不然也不行啊。 ---------------------------------------- 前一陣遇到在作.net Web開發時,碰到一個很奇怪的問題,就是Url中,若是將中文字符做爲參數值傳遞時,QueryString獲得的值可能會出錯。簡單的說,好比下面這個Url: UrlParmTest.aspx?parm1=中國&parm2=中國人 在Request.QueryString時,parm1和parm2獲得都是"中國",顯然出現了問題,但是在某些狀況下倒是正常的。 若是請求不是直接經過URL,而使用Response.Redirect在服務器端操做,沒有遇到過相似的問題。 當時我想中文是雙字節編碼,可能傳遞的時候就是有不肯定性,仍是用英文好。 但是爲何在Server端Redirect就是正常的,問題在哪裏呢? : 若是在.cs文件中設置中文參數,請在中文參數外使用Server.UrlEncode("中文")對中文進行Encode 若是在.aspx文件中設置,請使用<%=Server.UrlEncode("中文")%>進行Encode 在QueryString時,不用再進行Decode,能夠得到正常的中文字符串 下面是給出的一些解釋: UrlEncode把一些多字節字符轉換成url裏容許的單字節字符,原本瀏覽器就會自動作的,可是目前確實存在一些問題,因此本身再Encode一下,在接受端會自動對Url進行Decode。 我想Response.Redirect可能能夠確保做Encode的工做,因此沒有問題。 JavaScript Base64編碼和解碼,實現URL參數傳遞 爲何須要對參數進行編碼?相信有過開發的經驗的廣大程序員都知道,在Web中,如果直接在Url地址上傳遞參數值,如果中文,或者+等什麼的就會出現亂碼現象,如果數字或者英文的好象沒有什麼問題,簡言之,傳遞過來的參數是須要進行編碼的。 在這裏,也許有人會說,爲何不直接用Server.UrlDecode和Server.UrlEncode這兩個來進行編碼和解碼的操做呢? 的確,這兩個服務器端對象很好使用,用起來也很方便,可是,若在客戶端是HTML的Input,查詢的時候頁面是HTML或者其餘的,反正不是.NET的,那這個對象還能夠用嗎? 我如今就遇到這樣的問題,查詢的東東放在頁面,並且那個頁面我根本不想讓他是.aspx結尾的,哈,感受HTML的挺不錯,並且裏面的控件也是用HTML對象的。 下面先來看兩個函數,UTF16轉UTF8和UTF8轉Utf16的。 function utf16to8(str) { var out, i, len, c; out = ""; len = str.length; for(i = 0; i < len; i++) { c = str.charCodeAt(i); if ((c >= 0x0001) && (c <= 0x007F)) { out += str.charAt(i); } else if (c > 0x07FF) { out += String.fromCharCode(0xE0 | ((c >> 12) & 0x0F)); out += String.fromCharCode(0x80 | ((c >> 6) & 0x3F)); out += String.fromCharCode(0x80 | ((c >> 0) & 0x3F)); } else { out += String.fromCharCode(0xC0 | ((c >> 6) & 0x1F)); out += String.fromCharCode(0x80 | ((c >> 0) & 0x3F)); } } return out; } function utf8to16(str) { var out, i, len, c; var char2, char3; out = ""; len = str.length; i = 0; while(i < len) { c = str.charCodeAt(i++); switch(c >> 4) { case 0: case 1: case 2: case 3: case 4: case 5: case 6: case 7: // 0xxxxxxx out += str.charAt(i-1); break; case 12: case 13: // 110x xxxx 10xx xxxx char2 = str.charCodeAt(i++); out += String.fromCharCode(((c & 0x1F) << 6) | (char2 & 0x3F)); break; case 14: // 1110 xxxx 10xx xxxx 10xx xxxx char2 = str.charCodeAt(i++); char3 = str.charCodeAt(i++); out += String.fromCharCode(((c & 0x0F) << 12) | ((char2 & 0x3F) << 6) | ((char3 & 0x3F) << 0)); break; } } return out; } 那麼爲何須要進行轉化呢?由於在JavaScript中得到的中文字符是用UTF16進行編碼的,和咱們統一的頁面標準格式UTF-8可不同哦,因此須要先進行轉化,上面的函數UTF-16到UTF8,而後再進行Base64的編碼。 下面是關於Js進行Base64編碼和解碼的相關操做: var base64EncodeChars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; var base64DecodeChars = new Array( -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 62, -1, -1, -1, 63, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, -1, -1, -1, -1, -1, -1, -1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, -1, -1, -1, -1, -1, -1, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, -1, -1, -1, -1, -1); //客戶端Base64編碼 function base64encode(str) { var out, i, len; var c1, c2, c3; len = str.length; i = 0; out = ""; while(i < len) { c1 = str.charCodeAt(i++) & 0xff; if(i == len) { out += base64EncodeChars.charAt(c1 >> 2); out += base64EncodeChars.charAt((c1 & 0x3) << 4); out += "=="; break; } c2 = str.charCodeAt(i++); if(i == len) { out += base64EncodeChars.charAt(c1 >> 2); out += base64EncodeChars.charAt(((c1 & 0x3)<< 4) | ((c2 & 0xF0) >> 4)); out += base64EncodeChars.charAt((c2 & 0xF) << 2); out += "="; break; } c3 = str.charCodeAt(i++); out += base64EncodeChars.charAt(c1 >> 2); out += base64EncodeChars.charAt(((c1 & 0x3)<< 4) | ((c2 & 0xF0) >> 4)); out += base64EncodeChars.charAt(((c2 & 0xF) << 2) | ((c3 & 0xC0) >>6)); out += base64EncodeChars.charAt(c3 & 0x3F); } return out; } //客戶端Base64解碼 function base64decode(str) { var c1, c2, c3, c4; var i, len, out; len = str.length; i = 0; out = ""; while(i < len) { /* c1 */ do { c1 = base64DecodeChars[str.charCodeAt(i++) & 0xff]; } while(i < len && c1 == -1); if(c1 == -1) break; /* c2 */ do { c2 = base64DecodeChars[str.charCodeAt(i++) & 0xff]; } while(i < len && c2 == -1); if(c2 == -1) break; out += String.fromCharCode((c1 << 2) | ((c2 & 0x30) >> 4)); /* c3 */ do { c3 = str.charCodeAt(i++) & 0xff; if(c3 == 61) return out; c3 = base64DecodeChars[c3]; } while(i < len && c3 == -1); if(c3 == -1) break; out += String.fromCharCode(((c2 & 0XF) << 4) | ((c3 & 0x3C) >> 2)); /* c4 */ do { c4 = str.charCodeAt(i++) & 0xff; if(c4 == 61) return out; c4 = base64DecodeChars[c4]; } while(i < len && c4 == -1); if(c4 == -1) break; out += String.fromCharCode(((c3 & 0x03) << 6) | c4); } return out; } 這樣傳遞過去的值就能夠在服務器端解碼操做了。 下面是C#的Base64加碼和解碼的相關類: using System; using System.Data; using System.Configuration; using System.Web; using System.Web.Security; using System.Web.UI; using System.Web.UI.WebControls; using System.Web.UI.WebControls.WebParts; using System.Web.UI.HtmlControls; namespace CNVP.Base64 { /// <summary> /// MyBase64 的摘要說明 /// </summary> public class MyBase64 { public MyBase64() { // // TODO: 在此處添加構造函數邏輯 // } /// <summary> /// 服務器端Base64編碼 /// </summary> /// <param name="data"></param> /// <returns></returns> public string base64Encode(string data) { try { byte[] encData_byte = new byte[data.Length]; encData_byte = System.Text.Encoding.UTF8.GetBytes(data); string encodedData = Convert.ToBase64String(encData_byte); return encodedData; } catch (Exception e) { throw new Exception("Error in base64Encode" + e.Message); } } /// <summary> /// 服務器端Base64解碼 /// </summary> /// <param name="data"></param> /// <returns></returns> public string base64Decode(string data) { try { System.Text.UTF8Encoding encoder = new System.Text.UTF8Encoding(); System.Text.Decoder utf8Decode = encoder.GetDecoder(); byte[] todecode_byte = Convert.FromBase64String(data); int charCount = utf8Decode.GetCharCount(todecode_byte, 0, todecode_byte.Length); char[] decoded_char = new char[charCount]; utf8Decode.GetChars(todecode_byte, 0, todecode_byte.Length, decoded_char, 0); string result = new String(decoded_char); return result; } catch (Exception e) { throw new Exception("Error in base64Decode" + e.Message); } } } } var Keyword=base64encode(utf16to8(document.all.Keyword.value)); Keyword=Keyword.replace("+","%2B");//替換+,不然在服務器解碼的時候會出錯 服務器端使用如下代碼調用: CNVP.Base64.MyBase64 base64 = new CNVP.Base64.MyBase64(); Keyword=base64.base64Decode(Keyword);