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. }
### 回答1: `matcher.appendTail()` 是 Java `Matcher` 的一个方法,它将当前匹配的子字符串和其后面的字符串添加到一个 `StringBuffer` 对象,并返回最终的字符串。这个方法通常与 `Matcher.replaceAll()` 方法一起使用,用于替换匹配的字符串。具体用法可以参考 Java 官方文档的说明:https://docs.oracle.com/javase/8/docs/api/java/util/regex/Matcher.html#appendTail-java.lang.StringBuffer- ### 回答2: matcher.appendtail() 是 Java Matcher 的一个方法,用于将匹配操作的剩余部分追加到一个字符串缓冲区。 在 Java Matcher 用于对字符串进行匹配操作。当我们使用 Matcher 的 find() 方法找到一个匹配项之后,通常会使用 appendReplacement() 方法将匹配之前的部分替换为新的字符串。而 appendtail() 方法则正好相反,它用于将剩余的未匹配部分追加到一个字符串缓冲区,而不进行替换操作。 例如,假设我们有一个字符串 "Hello, World!",我们想要提取其的单词并打印出来。我们可以使用以下代码来达到目的: ```java import java.util.regex.Matcher; import java.util.regex.Pattern; public class Main { public static void main(String[] args) { String text = "Hello, World!"; Pattern pattern = Pattern.compile("\\w+"); // 匹配一个或多个字母或数字 Matcher matcher = pattern.matcher(text); StringBuilder result = new StringBuilder(); while (matcher.find()) { result.append(matcher.group()).append(" "); // 将匹配的单词追加到 result } matcher.appendTail(result); // 将剩余的未匹配部分追加到 result System.out.println(result.toString()); // 打印提取到的单词 } } ``` 运行这段代码,我们会得到输出结果 "Hello World ",其每个单词用空格隔开。可以看到,matcher.appendtail() 方法将未匹配部分的 "!" 追加到了 result 字符串缓冲区的末尾。 总结起来,matcher.appendtail() 方法是用于将 Matcher 对象剩余的未匹配部分追加到一个字符串缓冲区的方法。 ### 回答3: matcher.appendTail()是Java正则表达式的一个方法,它用于将剩余的字符串追加到StringBuffer对象。 在使用Java正则表达式进行匹配时,通常会使用到Matcher对象。Matcher对象提供了一系列的方法来进行匹配和操作字符串。其appendTail()方法就是其之一。 当我们在使用Matcher对象进行匹配时,可能只需要获取到部分匹配的结果,而不需要全部的匹配结果。此时,matcher.appendTail()方法就可以帮助我们将剩余的字符串追加到一个StringBuffer对象。 具体的使用方法是:首先创建一个StringBuffer对象,然后使用matcher.appendTail()方法将剩余的字符串追加到该对象。最后,通过调用StringBuffer对象的toString()方法将追加的结果转换为一个字符串。 该方法的主要作用是在进行匹配操作后,将未匹配的部分保存到一个StringBuffer对象,以便后续的操作。这样可以避免丢失未匹配的部分。 总之,matcher.appendTail()方法是在使用Matcher对象进行匹配时的一个非常有用的方法,可以帮助我们将未匹配的部分保存起来,以便后续的处理。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值