從此不再懼怕URI編碼 JavaScript及C# URI編碼詳解
更新時間:2012年03月07日 07:50:26 作者:
JavaScript中的還好,只提供了三個,C#中主要用的就有這么多,還沒有列出其他編碼(HTML),一多就弄不明白,弄不明白就心生恐懼,心生恐懼就變得苦逼,本文就向大家詳細(xì)解釋在JavaScript及C#中如何對URI進(jìn)行編碼的方法(注:本文不涉及到其他編碼)。
混亂的URI編碼
JavaScript中編碼有三種方法:escape、encodeURI、encodeURIComponent
C#中編碼主要方法:HttpUtility.UrlEncode、Server.UrlEncode、Uri.EscapeUriString、Uri.EscapeDataString
JavaScript中的還好,只提供了三個,C#中主要用的就有這么多,還沒有列出其他編碼(HTML),一多就弄不明白,弄不明白就心生恐懼,心生恐懼就變得苦逼,本文就向大家詳細(xì)解釋在JavaScript及C#中如何對URI進(jìn)行編碼的方法(注:本文不涉及到其他編碼)。
escape:不推薦使用
原因:eacape是BOM中的方法,只能對ASCII符號正確編碼,而encodeURI、encodeURIComponent可以對所有的Unicode符號編碼。ECMAScript v3 反對使用該方法,應(yīng)用使用 decodeURI() 和 decodeURIComponent() 替代它。
escape不編碼字符有69個:*,+,-,.,/,@,_,0-9,a-z,A-Z
encodeURI:用于對網(wǎng)址編碼(不包含參數(shù))
encodeURI不編碼字符有82個:!,#,$,&,',(,),*,+,,,-,.,/,:,;,=,?,@,_,~,0-9,a-z,A-Z
encodeURI就是為這個而設(shè)計的。encodeURI不對URI中的特殊字符進(jìn)行編碼,如冒號(:)、斜杠(/)。下面看個示例:
encodeURI("http://chabaoo.cn/a file with spaces.html")
// outputs http://chabaoo.cn/a%20file%20with%20spaces.html
可以看到僅僅把空格替換成了20%,所以此方法可用于對網(wǎng)址進(jìn)行編碼。
由于encodeURI不對冒號(:)、斜杠(/)進(jìn)行編碼,所以如果參數(shù)(如把網(wǎng)址作為參數(shù))中包含冒號(:)、斜杠(/),就會解析出錯,所以此方法不能對參數(shù)進(jìn)行編碼。
encodeURIComponent:用于對網(wǎng)址參數(shù)進(jìn)行編碼
encodeURIComponent不編碼字符有71個:!, ',(,),*,-,.,_,~,0-9,a-z,A-Z
可以看到此方法對:/都進(jìn)行了編碼,所以不能用它來對網(wǎng)址進(jìn)行編碼。由于此方法對中文,空格,井號(#),斜線(/),冒號(:)都進(jìn)行了編碼,所以適合對URI中的參數(shù)進(jìn)行編碼。看下面的示例:
var param="博客園";
var url="http://chabaoo.cn/?key="+encodeURIComponent(param)+"&page=1";
console.log(url);//outputs http://chabaoo.cn/?key=%E5%8D%9A%E5%AE%A2%E5%9B%AD&page=1
可以看到,這正是我們想要的結(jié)果(這里只對需要編碼的參數(shù)(page=1不需要編碼)進(jìn)行了編碼)。
Server.UrlEncode && HttpUtility.UrlEncode:不推薦
把這兩個放到一起說是因為這兩個方法在絕大多數(shù)情況下是一樣的。它們的區(qū)別是HttpUtility.UrlEncode默認(rèn)使用UTF8格式編碼,而Server.UrlEncode是使用系統(tǒng)預(yù)設(shè)格式編碼,Server.UrlEncode使用系統(tǒng)預(yù)設(shè)編碼做為參數(shù)調(diào)用HttpUtility.UrlEncode編碼,所以如果系統(tǒng)全局都用UTF8格式編碼,這兩個方法就是一樣的。
這兩個方法是怎么編碼的呢,我們來看個示例:
string url1 = "http://chabaoo.cn/a file with spaces.html?a=1&b=博客園#abc";
Response.Write(HttpUtility.UrlEncode(url1) );
//output
http%3a%2f%2fchabaoo.cn%2fa+file+with+spaces.html%3fa%3d1%26b%3d%e5%8d%9a%e5%ae%a2%e5%9b%ad%23abc
由上面的例子我們可以看出,HttpUtility.UrlEncode對冒號(:)和斜杠(/)進(jìn)行了編碼,所以不能用來對網(wǎng)址進(jìn)行編碼。
那么能不能對參數(shù)進(jìn)行編碼呢,答案也是否定的。因為在參數(shù)中空格應(yīng)該被編碼為%20而不是被HttpUtility.UrlEncode編碼為加號(+),所以不推薦用這兩個方法對URI進(jìn)行編碼。
Uri.EscapeUriString:用于對網(wǎng)址編碼(不包含參數(shù))
我們還是用例子說話:
string url1 = "http://chabaoo.cn/a file with spaces.html?a=1&b=博客園#abc";
Response.Write( Uri.EscapeUriString(url1));
//outputs:
http://chabaoo.cn/a%20file%20with%20spaces.html?a=1&b=%E5%8D%9A%E5%AE%A2%E5%9B%AD#abc
可以看出,Uri.EscapeUriString對空格進(jìn)行了編碼,也對中文進(jìn)行了編碼,但對冒號(:)、斜杠(/)和井號(#)未編碼,所以此方法可以用于網(wǎng)址進(jìn)行編碼,但不能對參數(shù)進(jìn)行編碼,作用類似JavaScript中的encodeURI方法。
Uri.EscapeDataString:用于對網(wǎng)址參數(shù)進(jìn)行編碼
仍然用例子說話:
string url1 = "http://chabaoo.cn/a file with spaces.html?a=1&b=博客園#abc";
Response.Write(Uri.EscapeDataString(url1));
//outputs:
http%3A%2F%2Fchabaoo.cn%2Fa%20file%20with%20spaces.html%3Fa%3D1%26b%3D%E5%8D%9A%E5%AE%A2%E5%9B%AD%23abc
可以看出,Uri.EscapeDataString對冒號(:)、斜杠(/)、空格、中文、井號(#)都進(jìn)行了編碼,所以此方法不可以用于網(wǎng)址進(jìn)行編碼,但可以用于對參數(shù)進(jìn)行編碼,作用類似JavaScript中的encodeURIComponent方法。
小結(jié)
在JavaScript中推薦的做法是用encodeURI對URI的網(wǎng)址部分編碼,用encodeURIComponent對URI中傳遞的參數(shù)進(jìn)行編碼。
在C#中推薦的做法是用Uri.EscapeUriString對URI的網(wǎng)址部分編碼,用Uri.EscapeDataString對URI中傳遞的參數(shù)進(jìn)行編碼。
解碼部分就不說了,與編碼方法相對應(yīng)。
作者:天行健,自強(qiáng)不息
出處:http://artwl.cnblogs.com
JavaScript中編碼有三種方法:escape、encodeURI、encodeURIComponent
C#中編碼主要方法:HttpUtility.UrlEncode、Server.UrlEncode、Uri.EscapeUriString、Uri.EscapeDataString
JavaScript中的還好,只提供了三個,C#中主要用的就有這么多,還沒有列出其他編碼(HTML),一多就弄不明白,弄不明白就心生恐懼,心生恐懼就變得苦逼,本文就向大家詳細(xì)解釋在JavaScript及C#中如何對URI進(jìn)行編碼的方法(注:本文不涉及到其他編碼)。
escape:不推薦使用
原因:eacape是BOM中的方法,只能對ASCII符號正確編碼,而encodeURI、encodeURIComponent可以對所有的Unicode符號編碼。ECMAScript v3 反對使用該方法,應(yīng)用使用 decodeURI() 和 decodeURIComponent() 替代它。
escape不編碼字符有69個:*,+,-,.,/,@,_,0-9,a-z,A-Z
encodeURI:用于對網(wǎng)址編碼(不包含參數(shù))
encodeURI不編碼字符有82個:!,#,$,&,',(,),*,+,,,-,.,/,:,;,=,?,@,_,~,0-9,a-z,A-Z
encodeURI就是為這個而設(shè)計的。encodeURI不對URI中的特殊字符進(jìn)行編碼,如冒號(:)、斜杠(/)。下面看個示例:
復(fù)制代碼 代碼如下:
encodeURI("http://chabaoo.cn/a file with spaces.html")
// outputs http://chabaoo.cn/a%20file%20with%20spaces.html
可以看到僅僅把空格替換成了20%,所以此方法可用于對網(wǎng)址進(jìn)行編碼。
由于encodeURI不對冒號(:)、斜杠(/)進(jìn)行編碼,所以如果參數(shù)(如把網(wǎng)址作為參數(shù))中包含冒號(:)、斜杠(/),就會解析出錯,所以此方法不能對參數(shù)進(jìn)行編碼。
encodeURIComponent:用于對網(wǎng)址參數(shù)進(jìn)行編碼
encodeURIComponent不編碼字符有71個:!, ',(,),*,-,.,_,~,0-9,a-z,A-Z
可以看到此方法對:/都進(jìn)行了編碼,所以不能用它來對網(wǎng)址進(jìn)行編碼。由于此方法對中文,空格,井號(#),斜線(/),冒號(:)都進(jìn)行了編碼,所以適合對URI中的參數(shù)進(jìn)行編碼。看下面的示例:
復(fù)制代碼 代碼如下:
var param="博客園";
var url="http://chabaoo.cn/?key="+encodeURIComponent(param)+"&page=1";
console.log(url);//outputs http://chabaoo.cn/?key=%E5%8D%9A%E5%AE%A2%E5%9B%AD&page=1
可以看到,這正是我們想要的結(jié)果(這里只對需要編碼的參數(shù)(page=1不需要編碼)進(jìn)行了編碼)。
Server.UrlEncode && HttpUtility.UrlEncode:不推薦
把這兩個放到一起說是因為這兩個方法在絕大多數(shù)情況下是一樣的。它們的區(qū)別是HttpUtility.UrlEncode默認(rèn)使用UTF8格式編碼,而Server.UrlEncode是使用系統(tǒng)預(yù)設(shè)格式編碼,Server.UrlEncode使用系統(tǒng)預(yù)設(shè)編碼做為參數(shù)調(diào)用HttpUtility.UrlEncode編碼,所以如果系統(tǒng)全局都用UTF8格式編碼,這兩個方法就是一樣的。
這兩個方法是怎么編碼的呢,我們來看個示例:
復(fù)制代碼 代碼如下:
string url1 = "http://chabaoo.cn/a file with spaces.html?a=1&b=博客園#abc";
Response.Write(HttpUtility.UrlEncode(url1) );
//output
http%3a%2f%2fchabaoo.cn%2fa+file+with+spaces.html%3fa%3d1%26b%3d%e5%8d%9a%e5%ae%a2%e5%9b%ad%23abc
由上面的例子我們可以看出,HttpUtility.UrlEncode對冒號(:)和斜杠(/)進(jìn)行了編碼,所以不能用來對網(wǎng)址進(jìn)行編碼。
那么能不能對參數(shù)進(jìn)行編碼呢,答案也是否定的。因為在參數(shù)中空格應(yīng)該被編碼為%20而不是被HttpUtility.UrlEncode編碼為加號(+),所以不推薦用這兩個方法對URI進(jìn)行編碼。
Uri.EscapeUriString:用于對網(wǎng)址編碼(不包含參數(shù))
我們還是用例子說話:
復(fù)制代碼 代碼如下:
string url1 = "http://chabaoo.cn/a file with spaces.html?a=1&b=博客園#abc";
Response.Write( Uri.EscapeUriString(url1));
//outputs:
http://chabaoo.cn/a%20file%20with%20spaces.html?a=1&b=%E5%8D%9A%E5%AE%A2%E5%9B%AD#abc
可以看出,Uri.EscapeUriString對空格進(jìn)行了編碼,也對中文進(jìn)行了編碼,但對冒號(:)、斜杠(/)和井號(#)未編碼,所以此方法可以用于網(wǎng)址進(jìn)行編碼,但不能對參數(shù)進(jìn)行編碼,作用類似JavaScript中的encodeURI方法。
Uri.EscapeDataString:用于對網(wǎng)址參數(shù)進(jìn)行編碼
仍然用例子說話:
復(fù)制代碼 代碼如下:
string url1 = "http://chabaoo.cn/a file with spaces.html?a=1&b=博客園#abc";
Response.Write(Uri.EscapeDataString(url1));
//outputs:
http%3A%2F%2Fchabaoo.cn%2Fa%20file%20with%20spaces.html%3Fa%3D1%26b%3D%E5%8D%9A%E5%AE%A2%E5%9B%AD%23abc
可以看出,Uri.EscapeDataString對冒號(:)、斜杠(/)、空格、中文、井號(#)都進(jìn)行了編碼,所以此方法不可以用于網(wǎng)址進(jìn)行編碼,但可以用于對參數(shù)進(jìn)行編碼,作用類似JavaScript中的encodeURIComponent方法。
小結(jié)
在JavaScript中推薦的做法是用encodeURI對URI的網(wǎng)址部分編碼,用encodeURIComponent對URI中傳遞的參數(shù)進(jìn)行編碼。
在C#中推薦的做法是用Uri.EscapeUriString對URI的網(wǎng)址部分編碼,用Uri.EscapeDataString對URI中傳遞的參數(shù)進(jìn)行編碼。
解碼部分就不說了,與編碼方法相對應(yīng)。
作者:天行健,自強(qiáng)不息
出處:http://artwl.cnblogs.com
您可能感興趣的文章:
相關(guān)文章
Git恢復(fù)之前版本的兩種方法reset、revert(圖文詳解)
這篇文章主要介紹了Git恢復(fù)之前版本的兩種方法reset、revert(圖文詳解),文中通過圖文示例介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07textarea 在IE和FF下?lián)Q行無法正常顯示的解決方法
今天在做項目時用到textarea 用戶輸入信息后顯示不換行在IE下測試成功在FF沒反應(yīng)2010-07-07教你如何在WordPress發(fā)布文章時自定義文章作者名稱
這篇文章主要介紹了如何在WordPress發(fā)布文章時自定義文章作者名稱2021-09-09VSCode設(shè)置默認(rèn)瀏覽器打開的兩種方式
在使用vscode編輯器的時候,如果我們不設(shè)置默認(rèn)瀏覽器,那么直接打開的是IE瀏覽器,在進(jìn)行頁面調(diào)試的時候無疑是有點麻煩的,這篇文章主要給大家介紹了關(guān)于VSCode設(shè)置默認(rèn)瀏覽器打開的兩種方式,需要的朋友可以參考下2023-09-09HTTP協(xié)議詳解_動力節(jié)點Java學(xué)院整理
這篇文章主要介紹了HTTP協(xié)議詳解,超文本傳輸協(xié)議(HTTP)是一種通信協(xié)議,它允許將超文本標(biāo)記語言(HTML)文檔從Web服務(wù)器傳送到客戶端的瀏覽器2017-07-07