1基础-013-1:正则表达式:string.replaceAll()中的特殊字符($ \)与matcher.appendReplacement

前端之家收集整理的这篇文章主要介绍了1基础-013-1:正则表达式:string.replaceAll()中的特殊字符($ \)与matcher.appendReplacement前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

string.replaceAll中的特殊字符@H_301_2@

string.replaceAll(String regex,String replacement)中的replacement参数即替换内容中含有特殊字符 $ \ 时,需转义。@H_301_2@

Java代码
  1. /*@H_301_2@@H_301_2@
  2. *字符串"$\"中的$与\字符互换位置@H_301_2@@H_301_2@
  3. */@H_301_2@@H_301_2@@H_301_2@
  4. public@H_301_2@@H_301_2@class@H_301_2@SpecialCharReplace{@H_301_2@@H_301_2@
  5. static@H_301_2@@H_301_2@void@H_301_2@main(String[]args){@H_301_2@@H_301_2@
  6. Stringstr="$\\"@H_301_2@;@H_301_2@@H_301_2@
  7. *string.replaceAll()中的特殊字符$与\@H_301_2@@H_301_2@
  8. *@H_301_2@@H_301_2@
  9. *由于$字符在作为替换内容时,是一个特殊字符,指反向引用前面的分组内容,所以把@H_301_2@@H_301_2@
  10. *某字符替换成$字符时,因该在前面加上转义字符\。@H_301_2@@H_301_2@
  11. *\字符就不用说了,本身就是转义字符,但为什么在作为替换内容时要使用四个\字符@H_301_2@@H_301_2@
  12. *,这里又不是用在正则表达式里?这就是因为\字符在作为替换内容里也是一个特殊字@H_301_2@@H_301_2@
  13. *符,它用来将前面讲的$字符进行转换的,所以也为特殊字符。以下是replaceAll的@H_301_2@@H_301_2@
  14. *源码片断,从源码就可以看出\$是两个特殊字符@H_301_2@@H_301_2@
  15. *if(nextChar=='\\'){@H_301_2@@H_301_2@
  16. *cursor++;@H_301_2@@H_301_2@
  17. *nextChar=replacement.charAt(cursor);@H_301_2@@H_301_2@
  18. *result.append(nextChar);@H_301_2@@H_301_2@
  19. *}elseif(nextChar=='$'){@H_301_2@@H_301_2@
  20. *//Skippast$@H_301_2@@H_301_2@
  21. *...@H_301_2@@H_301_2@
  22. *}else{@H_301_2@@H_301_2@
  23. *}@H_301_2@@H_301_2@
  24. System.out.println(str.replaceAll("\\$(\\W)\\\\","@H_301_2@\\\\$@H_301_2@1@H_301_2@\\$"));@H_301_2@//\$@H_301_2@@H_301_2@@H_301_2@
  25. }@H_301_2@
  26. @H_301_2@
  27. }@H_301_2@

Matcher对象的appendReplacement典型应用与特殊字符&\的进一步分析
@H_301_2@

问题的提出@H_301_2@@H_301_2@
@H_301_2@

字符串模板:
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中找不到域值,以空字符串""替换。@H_301_2@

问题的解决@H_301_2@@H_301_2@

class@H_301_2@RegexExam{@H_301_2@@H_301_2@
  • void@H_301_2@main(Stringargs[]){@H_301_2@@H_301_2@
  • HashMapdata=new@H_301_2@HashMap();@H_301_2@@H_301_2@
  • Stringtemplate="尊敬的客户${customerName}你好!本次消费金额${amount},"@H_301_2@@H_301_2@@H_301_2@
  • +"您帐户${accountNumber}上的余额为${balance},欢迎下次光临!"@H_301_2@;@H_301_2@@H_301_2@
  • data.put("customerName"@H_301_2@,@H_301_2@"刘明"@H_301_2@);@H_301_2@@H_301_2@
  • data.put("accountNumber"@H_301_2@,@H_301_2@"888888888"@H_301_2@);@H_301_2@@H_301_2@
  • data.put("balance"@H_301_2@,@H_301_2@"$1000000.00"@H_301_2@);@H_301_2@@H_301_2@
  • data.put("amount"@H_301_2@,@H_301_2@"$1000.00"@H_301_2@);@H_301_2@@H_301_2@
  • try@H_301_2@{@H_301_2@@H_301_2@
  • System.out.println(composeMessage(template,data));@H_301_2@
  • }catch@H_301_2@(Exceptione){@H_301_2@@H_301_2@
  • e.printStackTrace();@H_301_2@
  • static@H_301_2@StringcomposeMessage(Stringtemplate,Mapdata)@H_301_2@@H_301_2@
  • throws@H_301_2@Exception{@H_301_2@@H_301_2@
  • Stringregex="\\$\\{(.+?)\\}"@H_301_2@;@H_301_2@@H_301_2@
  • Patternpattern=Pattern.compile(regex);@H_301_2@
  • Matchermatcher=pattern.matcher(template);@H_301_2@
  • *sb用来存储替换过的内容,它会把多次处理过的字符串按源字符串序@H_301_2@@H_301_2@
  • *存储起来。@H_301_2@@H_301_2@
  • StringBuffersb=new@H_301_2@StringBuffer();@H_301_2@@H_301_2@
  • while@H_301_2@(matcher.find()){@H_301_2@@H_301_2@
  • Stringname=matcher.group(1@H_301_2@);@H_301_2@//键名@H_301_2@@H_301_2@@H_301_2@
  • Stringvalue=(String)data.get(name);//键值@H_301_2@@H_301_2@@H_301_2@
  • if@H_301_2@(value==@H_301_2@null@H_301_2@){@H_301_2@@H_301_2@
  • value=""@H_301_2@;@H_301_2@@H_301_2@
  • else@H_301_2@{@H_301_2@@H_301_2@
  • *由于$出现在replacement中时,表示对捕获组的反向引用,所以要对上面替换内容@H_301_2@@H_301_2@
  • *中的$进行替换,让它们变成"\$1000.00"或"\$1000000000.00",这样@H_301_2@@H_301_2@
  • *在下面使用matcher.appendReplacement(sb,value)进行替换时就不会把@H_301_2@@H_301_2@
  • *$1看成是对组的反向引用了,否则会使用子匹配项值amount或balance替换$1@H_301_2@@H_301_2@
  • *,最后会得到错误结果:@H_301_2@@H_301_2@
  • *尊敬的客户刘明你好!本次消费金额amount000.00,您帐户888888888上的余额@H_301_2@@H_301_2@
  • *为balance000000.00,欢迎下次光临!@H_301_2@@H_301_2@
  • *要把$替换成\$,则要使用\\\\\\&来替换,因为一个\要使用\\\来进@H_301_2@@H_301_2@
  • *行替换,而一个$要使用\\$来进行替换,因\与$在作为替换内容时都属于@H_301_2@@H_301_2@
  • *特殊字符:$字符表示反向引用组,而\字符又是用来转义$字符的。@H_301_2@@H_301_2@
  • value=value.replaceAll("\\$"@H_301_2@,@H_301_2@"\\\\\\$"@H_301_2@);@H_301_2@@H_301_2@
  • //System.out.println("value="+value);@H_301_2@@H_301_2@@H_301_2@
  • *经过上面的替换操作,现在的value中含有$特殊字符的内容被换成了"\$1000.00"@H_301_2@@H_301_2@
  • *或"\$1000000000.00"了,最后得到下正确的结果:@H_301_2@@H_301_2@
  • *尊敬的客户刘明你好!本次消费金额$1000.00,您帐户888888888上的@H_301_2@@H_301_2@
  • *余额为$1000000.00,欢迎下次光临!@H_301_2@@H_301_2@
  • *另外,我们在这里使用Matcher对象的appendReplacement()方法来进行替换操作,而@H_301_2@@H_301_2@
  • *不是使用String对象的replaceAll()或replaceFirst()方法来进行替换操作,因为@H_301_2@@H_301_2@
  • *它们都能只能进行一次性简单的替换操作,而且只能替换成一样的内容,而这里则是要求每@H_301_2@@H_301_2@
  • *一个匹配式的替换值都不同,所以就只能在循环里使用appendReplacement方式来进行逐@H_301_2@@H_301_2@
  • *个替换了。@H_301_2@@H_301_2@
  • matcher.appendReplacement(sb,value);@H_301_2@
  • System.out.println("sb="@H_301_2@+sb.toString());@H_301_2@@H_301_2@
  • //最后还得要把尾串接到已替换的内容后面去,这里尾串为“,欢迎下次光临!”@H_301_2@@H_301_2@@H_301_2@
  • matcher.appendTail(sb);@H_301_2@
  • return@H_301_2@sb.toString();@H_301_2@@H_301_2@
  • } @H_301_2@
  • 猜你在找的正则表达式相关文章