轉自:http://blog.csdn.net/hj7jay/article/details/51280405
衆所周知頁面上的字符內容一般都須要進行HTML轉義才能正確顯示,尤爲對於Input,Textarea提交的內容,更是要進行轉義以防止javascript注入攻擊。javascript
一般的HTML轉義主要是針對內容中的"<",">","&",以及空格、單雙引號等。但其實還有不少字符也須要進行轉義。具體的能夠參考
這篇文章。
** 一、HTML轉義
參考上面的提到的文章,基本上能夠肯定如下的轉義的範圍和方式。
1)對"\""、"&"、"'"、"<"、">"、空格(0x20)、0x00到0x20、0x7F-0xFF
以及0x0100-0x2700的字符進行轉義,基本上就覆蓋的比較全面了。
用javascript的正則表達式能夠寫爲:
this.REGX_HTML_ENCODE = /"|&|'|<|>|[\x00-\x20]|[\x7F-\xFF]|[\u0100-\u2700]/g;
2)爲保證轉義結果對瀏覽器的無差異,轉義編碼爲實體編號,而不用實體名稱。
3)空格(0x20)一般轉義爲「 」也就是「 」。
轉義的代碼很是簡單:
- this.encodeHtml = function(s){
- return (typeof s != "string") ? s :
- s.replace(this.REGX_HTML_ENCODE,
- function($0){
- var c = $0.charCodeAt(0), r = ["&#"];
- c = (c == 0x20) ? 0xA0 : c;
- r.push(c); r.push(";");
- return r.join("");
- });
- };
** 二、反轉義
既然有轉義,天然須要反轉義。
1) 對「&#num;」實體編號的轉義,直接提取編號而後fromCharCode就能夠獲得字符。
2) 對於諸如「<」,須要創建一張以下的表來查詢。
this.HTML_DECODE = {
"<" : "<",
">" : ">",
"&" : "&",
" ": " ",
""": "\"",
"©": "©"
// Add more
};
由此咱們能夠有反轉義的正則表達式:
this.REGX_HTML_DECODE = /&\w+;|&#(\d+);/g;
反轉的代碼也很簡單,以下:
- this.decodeHtml = function(s){
- return (typeof s != "string") ? s :
- s.replace(this.REGX_HTML_DECODE,
- function($0,$1){
- var c = this.HTML_ENCODE[$0]; // 嘗試查表
- if(c === undefined){
- // Maybe is Entity Number
- if(!isNaN($1)){
- c = String.fromCharCode(($1 == 160) ? 32 : $1);
- }else{
- // Not Entity Number
- c = $0;
- }
- }
- return c;
- });
- };
** 三、一個有意思的認識
其實在用正則表達式轉義以前,我一直都是用遍歷整個字符串,逐個比較字符的方式。直到有一天,看到一篇文章說,javascript正則表達式是C實現的,比本身用javascript遍歷字符要快,因而我就試着改寫成上面這種方式。雖然代碼看起來的確顯得神祕而又牛叉,但遺憾的是,在個人Chrome 11 (FreeBSD 64 9.0)上,遍歷字符轉義/反轉的方式要比上面正則表達式的代碼快2到3倍(字符串長度越長越明顯)。其實,想一想也能明白爲何。
** 四、完整版本的代碼
- $package("js.lang"); // 沒有包管理時,也可簡單寫成 js = {lang:{}};
-
- js.lang.String = function(){
-
- this.REGX_HTML_ENCODE = /"|&|'|<|>|[\x00-\x20]|[\x7F-\xFF]|[\u0100-\u2700]/g;
-
- this.REGX_HTML_DECODE = /&\w+;|&#(\d+);/g;
-
- this.REGX_TRIM = /(^\s*)|(\s*$)/g;
-
- this.HTML_DECODE = {
- "<" : "<",
- ">" : ">",
- "&" : "&",
- " ": " ",
- """: "\"",
- "©": ""
-
- // Add more
- };
-
- this.encodeHtml = function(s){
- s = (s != undefined) ? s : this.toString();
- return (typeof s != "string") ? s :
- s.replace(this.REGX_HTML_ENCODE,
- function($0){
- var c = $0.charCodeAt(0), r = ["&#"];
- c = (c == 0x20) ? 0xA0 : c;
- r.push(c); r.push(";");
- return r.join("");
- });
- };
-
- this.decodeHtml = function(s){
- var HTML_DECODE = this.HTML_DECODE;
-
- s = (s != undefined) ? s : this.toString();
- return (typeof s != "string") ? s :
- s.replace(this.REGX_HTML_DECODE,
- function($0, $1){
- var c = HTML_DECODE[$0];
- if(c == undefined){
- // Maybe is Entity Number
- if(!isNaN($1)){
- c = String.fromCharCode(($1 == 160) ? 32:$1);
- }else{
- c = $0;
- }
- }
- return c;
- });
- };
-
- this.trim = function(s){
- s = (s != undefined) ? s : this.toString();
- return (typeof s != "string") ? s :
- s.replace(this.REGX_TRIM, "");
- };
-
-
- this.hashCode = function(){
- var hash = this.__hash__, _char;
- if(hash == undefined || hash == 0){
- hash = 0;
- for (var i = 0, len=this.length; i < len; i++) {
- _char = this.charCodeAt(i);
- hash = 31*hash + _char;
- hash = hash & hash; // Convert to 32bit integer
- }
- hash = hash & 0x7fffffff;
- }
- this.__hash__ = hash;
-
- return this.__hash__;
- };
-
- };
-
- js.lang.String.call(js.lang.String);
在實際的使用中能夠有兩種方式:
1)使用js.lang.String.encodeHtml(s)和js.lang.String.decodeHtml(s)。
2)還能夠直接擴展String的prototype
- js.lang.String.call(String.prototype);
-
- // 那麼
-
- var str = "&'\"中國abc def";
-
- var ec_str = str.encodeHtml();
-
- document.write(ec_str);
-
- document.write(""); // CU的博客在線編輯有bug,
- 放不上來!!!
-
- var dc_str = ec_str.decodeHtml();
-
- document.write(dc_str);