字元串的replace()方法隱藏著什麼不可告人秘密?

来源:http://www.cnblogs.com/star91/archive/2016/06/21/5603670.html
-Advertisement-
Play Games

最近在做JS演算法項目時發現一個令我匪夷所思的問題, 這裡想記錄一下問題。 首先介紹一下字元串replace()方法的基本用法。 replace() 方法使用一個替換值(replacement)替換掉一個匹配模式(pattern)在原字元串中某些或所有的匹配項,並返回替換後的字元串。這個替換模式可以是 ...


最近在做JS演算法項目時發現一個令我匪夷所思的問題, 這裡想記錄一下問題。

首先介紹一下字元串replace()方法的基本用法。

replace() 方法使用一個替換值(replacement)替換掉一個匹配模式(pattern)在原字元串中某些或所有的匹配項,並返回替換後的字元串。這個替換模式可以是字元串或者RegExp(正則表達式),替換值可以是一個字元串或者一個函數。

語法EDIT

str.replace(regexp|substrnewSubStr|function[, flags])

參數

regexp
一個 RegExp 對象。該正則所匹配的內容會被第二個參數的返回值替換掉。
substr
一個要被 newSubStr 替換的字元串。
newSubStr
替換掉第一個參數在原字元串中的匹配部分。該字元串中可以內插一些特殊的變數名。
function

一個用來創建新子字元串的函數,該函數的返回值將替換掉第一個參數匹配到的結果。該函數的參數描述請參考 指定一個函數作為參數 小節。

返回值

一個部分或全部匹配由替代模式所取代的新的字元串。

關於這個方法具體的信息參考MDN再好不過了。

String.prototype.replace() - JavaScript | MDN

 

現在有一個非常簡單的需求:將HTML代碼中的特殊字元進行實體轉義:

先看一個簡單的輸出沒有錯誤的版本:

正確方法一:

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 function convertHTML(str) {   // &colon;&rpar;   var pattern=/[&<>"']/g;   return str.replace(pattern,function(match){     switch(match){         case "<":         return "&lt;";         case ">":         return "&gt;";         case "&":         return "&amp;";         case "\"":         return "&quot;";         case "\'":         return "&apos;";     }   }); } console.log(convertHTML("<Tom & Jerry>"));//&lt;Tom &amp; Jerry&gt;

指定一個函數作為第二個參數。在這種情況下,當匹配執行後, 該函數就會執行。 函數的返回值作為替換字元串。 (註意:  上面提到的特殊替換參數在這裡不能被使用。) 另外要註意的是, 如果第一個參數是正則表達式, 並且其為全局匹配模式, 那麼這個方法將被多次調用, 每次匹配都會被調用。

後來我對上述的代碼進行了改造,使代碼看起來更明瞭:

正確方法二:

 

1 2 3 4 5 6 7 8 9 10 11 12 13 function convertHTML(str) {   // &colon;&rpar;   var pattern=/[&<>"']/g;   rules={"&":"&amp;","<":"&lt;",">":"&gt;",'"':"&quot;","'":"&apos;"};  return str.replace(pattern,function(match){     return  rules[match];   });     } console.log(convertHTML("<Tom & Jerry>"));//&lt;Tom &amp; Jerry&gt;

很明顯非常簡單的一個改造,我將規則鍵值對放在了rules對象中,輸出結果同樣也是正確的。

接下來,我發現貌似使用RegExp構造函數屬性還能使代碼變得更加簡單。RegExp構造函數屬性如下表所示:

其中一個叫做lastMatch的屬性深深吸引了我。我在想能不能像下麵這樣改造一下:

錯誤方法:

1 2 3 4 5 6 function convertHTML(str) {   var pattern=/[&<>"']/g;   rules={"&":"&amp;","<":"&lt;",">":"&gt;",'"':"&quot;","'":"&apos;"};   return str.replace(pattern,rules[RegExp.lastMatch]); } console.log(convertHTML("<Tom & Jerry>"));

寫完這個代碼,突然覺得自己聰明絕世,飄飄然了呢,然而一瓢冷水潑下來了。運行結果如下:

 

 
1 undefinedTom undefined Jerryundefined

這是什麼鬼,貌似該替換的字元都被替換成了undefined。接下來我加入了一條測試語句:

 

 

 
1 2 3 4 5 6 7 8 9 function convertHTML(str) {   // &colon;&rpar;   var pattern=/[&<>"']/g;   rules={"&":"&amp;","<":"&lt;",">":"&gt;",'"':"&quot;","'":"&apos;"};

  var result=str.replace(pattern,rules[RegExp.lastMatch]);

//下麵是測試代碼

  console.log(RegExp.lastMatch);//>   return result; } console.log(convertHTML("<Tom & Jerry>"));

根據測試代碼,最後匹配的是“>”說明匹配是正確的,但是最後為什麼沒有按照規則進行替換呢,我想應該是replace()方法的實現機制我們沒有弄清楚,網上沒有找到相關的資料,所以只能提醒自己以後不能寫這樣的代碼了,在這種情況下,還是乖乖函數作為第二個參數。。。。。。等等,用函數做第二個參數,我又進行了下麵的修改:

1 2 3 4 5 6 7 8 9 10 function convertHTML(str) {   // &colon;&rpar;   var pattern=/[&<>"']/g;   rules={"&":"&amp;","<":"&lt;",">":"&gt;",'"':"&quot;","'":"&apos;"};   var result=str.replace(pattern,function(){     return rules[RegExp.lastMatch]   });   return result; } console.log(convertHTML("<Tom & Jerry>"));//&lt;Tom &amp; Jerry&gt;

註意上面我還是使用了正則表達式函數屬性與方法二是有區別的,但是結果卻是離奇的正確了。(難道閉包作怪)

 

 

replace()究竟隱藏著什麼秘密,還有待挖掘??

來自為知筆記(Wiz)




您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • CONTENT(目錄) 前言 Variable declearation:three rules you can break 1.Don't set var statement in a block 2.Don't set var statement in a loop 3.Set a variab ...
  • 前言: 1.HTML5的發展非常迅速,可以說已經是前端開發人員的標配,在電商類型的APP中更是運用廣泛,這個系列的文章是本人自己整理,儘量將開發中不常用到的剔除,將經常使用的拿出來,使需要的朋友能夠真正快速入門,如果有哪些不清楚的地方或者錯誤,歡迎聯繫我 2.更新時間沒有規律,一般會在3天左右更新一 ...
  • 在現在的很多頁面中,都運用到了百度地圖來定位,例如: 像這樣的地圖,我們可以通過手動來進行放大、縮小、移動等來查找具體的地址,特別方便,在頁面上引用也顯得頁面很有特點,那麼,應該怎麼樣來製作這種地圖呢? 一、獲取源代碼 網址:http://api.map.baidu.com/lbsapi/creat ...
  • 閉包(closure)是Javascript語言的一個難點,也是它的特色,很多高級應用都要依靠閉包實現。 一、變數的作用域 要理解閉包,首先必須理解Javascript特殊的變數作用域。 變數的作用域無非就是兩種:全局變數和局部變數。 Javascript語言的特殊之處,就在於函數內部可以直接讀取全 ...
  • 近幾年,人臉識別技術在身份認證領域的應用已經有了較多應用,例如:支付寶、招行的取款、養老金領取等方面,但在杜絕假冒、認證安全性等方面,目前還是一個比較需要進一步解決的課題,特別是在移動端的活體認證技術方面。 本文介紹了在HTML5 環境下可以採用clmtrackr.js 檢測工具,結合人臉模型,實現 ...
  • 1、日期格式轉時間戮 2、時間戮轉日期格式 ...
  • 經常我們看電影都會看到電影結束後會出現一段介紹這個電影的導演、編劇、主演、友情出演等等一些電影信息的滾動字幕,那麼那個效果可以用html+css+div實現。具體實現代碼如下: html代碼: 滾動標簽:<marquee>要滾動的文字/圖像<img src=””> </marquee> 下麵是我對滾 ...
  • 1,給第一人稱控制器添加腳本:playercollisions.js 腳本中只定義變數,先不添加方法: #pragma strict var door_open_time:float=3.0; var door_open_sound:AudioClip; var door_shut_sound:Au ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...