(轉)正則表達式:string.replaceAll()中的特殊字符($ \)與matcher.appendReplacement


string.replaceAll中的特殊字符

string.replaceAll(String regex, String replacement)中的replacement參數即替換內容中含有特殊字符 $ \ 時,需轉義。

 

Java代碼   收藏代碼
  1. /* 
  2.  * 字符串"$ \"中的$與\字符互換位置 
  3.  */  
  4. public class SpecialCharReplace {  
  5.     public static void main(String[] args) {  
  6.         String str = "$ \\";  
  7.         /* 
  8.          * string.replaceAll()中的特殊字符 $ 與 \  
  9.          *  
  10.          * 由於 $ 字符在作為替換內容時,是一個特殊字符,指反向引用前面的分組內容,所以把 
  11.          * 某字符替換成 $ 字符時,因該在前面加上轉義字符 \。 
  12.          * \ 字符就不用說了,本身就是轉義字符,但為什么在作為替換內容時要使用四個 \ 字符 
  13.          * ,這里又不是用在正則表達式里?這就是因為 \ 字符在作為替換內容里也是一個特殊字 
  14.          * 符,它用來將前面講的 $ 字符進行轉換的,所以也為特殊字符。以下是replaceAll的 
  15.          * 源碼片斷,從源碼就可以看出 \$ 是兩個特殊字符 
  16.          *  
  17.          * if (nextChar == '\\') { 
  18.          *      cursor++; 
  19.          *      nextChar = replacement.charAt(cursor); 
  20.          *      result.append(nextChar); 
  21.          *      cursor++; 
  22.          * } else if (nextChar == '$') { 
  23.          *      // Skip past $ 
  24.          *      cursor++; 
  25.          *      ... 
  26.          * }else { 
  27.          *      result.append(nextChar); 
  28.          *      cursor++; 
  29.          * } 
  30.          */  
  31.         System.out.println(str.replaceAll("\\$(\\W)\\\\", "\\\\$1\\$"));// \ $  
  32.     }  
  33.   
  34. }   

Matcher對象的appendReplacement典型應用與特殊字符&\的進一步分析

問題的提出 

字符串模板: 
    String template="尊敬的客戶${customerName}你好!本次消費金額${amount},您帳戶${accountNumber}上的余額為${balance},歡迎下次光臨!"; 
其中以 ${ 開始 } 結尾的為待替換的變量域。 
數據存放於Map中,key為域名,value為域值。如: 
Map-- 
    customerName = 劉明 
    accountNumber = 888888888
    balance = $1000000.00
    amount = $1000.00 
請編寫函數: 
    public static String composeMessage(String template, Map data) throw Exception 
實現將任意模板字符串中的變量域,按域名替換為data中的域值。 
例如,上例替換結果為: 
    "尊敬的客戶劉明你好!本次消費金額$1000.00,您帳戶888888888上的余額為$1000000.00,歡迎下次光臨!" 
注:如果Map中找不到域值,以空字符串""替換。 

問題的解決

Java代碼   收藏代碼
    1. public class RegexExam {  
    2.     public static void main(String args[]) {  
    3.         HashMap data = new HashMap();  
    4.         String template = "尊敬的客戶${customerName}你好!本次消費金額${amount},"  
    5.                 + "您帳戶${accountNumber}上的余額為${balance},歡迎下次光臨!";  
    6.         data.put("customerName", "劉明");  
    7.         data.put("accountNumber", "888888888");  
    8.         data.put("balance", "$1000000.00");  
    9.         data.put("amount", "$1000.00");  
    10.         try {  
    11.             System.out.println(composeMessage(template, data));  
    12.         } catch (Exception e) {  
    13.             e.printStackTrace();  
    14.         }  
    15.     }  
    16.   
    17.     public static String composeMessage(String template, Map data)  
    18.             throws Exception {  
    19.         String regex = "\\$\\{(.+?)\\}";  
    20.         Pattern pattern = Pattern.compile(regex);  
    21.         Matcher matcher = pattern.matcher(template);  
    22.         /* 
    23.          * sb用來存儲替換過的內容,它會把多次處理過的字符串按源字符串序 
    24.          * 存儲起來。 
    25.          */  
    26.         StringBuffer sb = new StringBuffer();  
    27.         while (matcher.find()) {  
    28.             String name = matcher.group(1);//鍵名  
    29.             String value = (String) data.get(name);//鍵值  
    30.             if (value == null) {  
    31.                 value = "";  
    32.             } else {  
    33.                 /* 
    34.                  * 由於$出現在replacement中時,表示對捕獲組的反向引用,所以要對上面替換內容 
    35.                  * 中的 $ 進行替換,讓它們變成 "\$1000.00" 或 "\$1000000000.00" ,這樣 
    36.                  * 在下面使用 matcher.appendReplacement(sb, value) 進行替換時就不會把 
    37.                  * $1 看成是對組的反向引用了,否則會使用子匹配項值amount 或 balance替換 $1 
    38.                  * ,最后會得到錯誤結果: 
    39.                  * 
    40.                  * 尊敬的客戶劉明你好!本次消費金額amount000.00,您帳戶888888888上的余額 
    41.                  * 為balance000000.00,歡迎下次光臨! 
    42.                  * 
    43.                  * 要把 $ 替換成 \$ ,則要使用 \\\\\\& 來替換,因為一個 \ 要使用 \\\ 來進 
    44.                  * 行替換,而一個 $ 要使用 \\$ 來進行替換,因 \ 與  $ 在作為替換內容時都屬於 
    45.                  * 特殊字符:$ 字符表示反向引用組,而 \ 字符又是用來轉義 $ 字符的。 
    46.                  */  
    47.                 value = value.replaceAll("\\$", "\\\\\\$");  
    48.                 //System.out.println("value=" + value);  
    49.             }  
    50.             /* 
    51.              * 經過上面的替換操作,現在的 value 中含有 $ 特殊字符的內容被換成了"\$1000.00" 
    52.              * 或 "\$1000000000.00" 了,最后得到下正確的結果: 
    53.              * 
    54.              * 尊敬的客戶劉明你好!本次消費金額$1000.00,您帳戶888888888上的 
    55.              * 余額為$1000000.00,歡迎下次光臨! 
    56.              * 
    57.              * 另外,我們在這里使用Matcher對象的appendReplacement()方法來進行替換操作,而 
    58.              * 不是使用String對象的replaceAll()或replaceFirst()方法來進行替換操作,因為 
    59.              * 它們都能只能進行一次性簡單的替換操作,而且只能替換成一樣的內容,而這里則是要求每 
    60.              * 一個匹配式的替換值都不同,所以就只能在循環里使用appendReplacement方式來進行逐 
    61.              * 個替換了。 
    62.              */  
    63.             matcher.appendReplacement(sb, value);  
    64.             System.out.println("sb = " + sb.toString());  
    65.         }  
    66.         //最后還得要把尾串接到已替換的內容后面去,這里尾串為“,歡迎下次光臨!”  
    67.         matcher.appendTail(sb);  
    68.         return sb.toString();  
    69.     }  
    70. }  


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM