这里面我们分析一下replace与replaceAll方法的差异以及原理。

replace各个方法的定义

一、replaceFirst方法

public String replaceFirst(String regex, String replacement) {

return Pattern.compile(regex).matcher(this).replaceFirst(replacement);

}

二、replace方法

public String replace(CharSequence target, CharSequence replacement) {

return Pattern.compile(target.toString(), Pattern.LITERAL).matcher(this).replaceAll(Matcher.quoteReplacement(replacement.toString()));

}

三、replaceAll方法

public String replaceAll(String regex, String replacement) {

return Pattern.compile(regex).matcher(this).replaceAll(replacement);

}

replace各个方法的原理

我们通过以下的例子来分析他们的原理。

@Test

public void stringReplace() {

replaceFirst("year = 1929. month=07, day=29, other=\\d{2}");

}

public void replaceFirst(String string) {

System.out.println(string.replaceFirst("\\d{2}", "--"));

System.out.println(string.replace("\\d{2}", "--"));

System.out.println(string.replace("29", "--"));

System.out.println(string.replaceAll("\\d{2}", "--"));

}

// year = --29. month=07, day=29, other=\d{2}

// year = 1929. month=07, day=29, other=--

// year = 19--. month=07, day=--, other=\d{2}

// year = ----. month=--, day=--, other=\d{2}

一、首先我们分析一下replaceFirst与replaceAll方法,他们的区别在于Pattern构建之后Matcher调用的方法不同。一个是reaplceFirst、一个是replaceAll方法。这两个方法现在可以分析一下。

1、首先对于Matcher的replceFirst方法:可以看到只调用一下的appendReplacement和appendTail方法。关于appendReplacement方法后面可以贴出源码,实现比较复杂

public String replaceFirst(String replacement) {

if (replacement == null)

throw new NullPointerException("replacement");

reset();

if (!find())

return text.toString();

StringBuffer sb = new StringBuffer();

appendReplacement(sb, replacement);

appendTail(sb);

return sb.toString();

}

2、对于Matcher的replceAll方法,和上述的replaceFirst方法类似。只不过是多次调用了appendReplacement的替换函数。直到没有匹配为止

public String replaceAll(String replacement) {

reset();

boolean result = find();

if (result) {

StringBuffer sb = new StringBuffer();

do {

appendReplacement(sb, replacement);

result = find();

} while (result);

appendTail(sb);

return sb.toString();

}

return text.toString();

}

二、对于replace方法,和上述的replaceAll方法主要有以下两种区别。

1、在Pattern.compile时,添加了Pattern.LITERAL的flag,表示pattern会把regex当作纯文本来处理了。比如\\d{2}不转义成两个0-9的数字,而是当作纯文本\\d{2}看待。

2、在调用MatcherMatcher.quoteReplacement(replacement.toString())方法对replacement做了对特殊符号($和\)作去除转义的操作。

public static String quoteReplacement(String s) {

if ((s.indexOf('\\') == -1) && (s.indexOf('$') == -1))

return s;

StringBuilder sb = new StringBuilder();

for (int i=0; i

char c = s.charAt(i);

if (c == '\\' || c == '$') {

sb.append('\\');

}

sb.append(c);

}

return sb.toString();

}

但是为何只对\\和$做处理呢?

三、以下是我们的重点appendReplacement方法

public Matcher appendReplacement(StringBuffer sb, String replacement) {

// If no match, return error

if (first < 0)

throw new IllegalStateException("No match available");

// Process substitution string to replace group references with groups

int cursor = 0;

StringBuilder result = new StringBuilder();

while (cursor < replacement.length()) {

char nextChar = replacement.charAt(cursor);

if (nextChar == '\\') {

cursor++;

if (cursor == replacement.length())

throw new IllegalArgumentException("character to be escaped is missing");

nextChar = replacement.charAt(cursor);

result.append(nextChar);

cursor++;

} else if (nextChar == '$') {

// Skip past $

cursor++;

// Throw IAE if this "$" is the last character in replacement

if (cursor == replacement.length())

throw new IllegalArgumentException("Illegal group reference: group index is missing");

nextChar = replacement.charAt(cursor);

int refNum = -1;

if (nextChar == '{') {

cursor++;

StringBuilder gsb = new StringBuilder();

while (cursor < replacement.length()) {

nextChar = replacement.charAt(cursor);

if (ASCII.isLower(nextChar) ||

ASCII.isUpper(nextChar) ||

ASCII.isDigit(nextChar)) {

gsb.append(nextChar);

cursor++;

} else {

break;

}

}

if (gsb.length() == 0)

throw new IllegalArgumentException("named capturing group has 0 length name");

if (nextChar != '}')

throw new IllegalArgumentException("named capturing group is missing trailing '}'");

String gname = gsb.toString();

if (ASCII.isDigit(gname.charAt(0)))

throw new IllegalArgumentException("capturing group name {" + gname + "} starts with digit character");

if (!parentPattern.namedGroups().containsKey(gname))

throw new IllegalArgumentException("No group with name {" + gname + "}");

refNum = parentPattern.namedGroups().get(gname);

cursor++;

} else {

// The first number is always a group

refNum = (int)nextChar - '0';

if ((refNum < 0)||(refNum > 9))

throw new IllegalArgumentException("Illegal group reference");

cursor++;

// Capture the largest legal group string

boolean done = false;

while (!done) {

if (cursor >= replacement.length()) {

break;

}

int nextDigit = replacement.charAt(cursor) - '0';

if ((nextDigit < 0)||(nextDigit > 9)) { // not a number

break;

}

int newRefNum = (refNum * 10) + nextDigit;

if (groupCount() < newRefNum) {

done = true;

} else {

refNum = newRefNum;

cursor++;

}

}

}

// Append group

if (start(refNum) != -1 && end(refNum) != -1)

result.append(text, start(refNum), end(refNum));

} else {

result.append(nextChar);

cursor++;

}

}

// Append the intervening text

sb.append(text, lastAppendPosition, first);

// Append the match substitution

sb.append(result);

lastAppendPosition = last;

return this;

}

四、以下是appendTail的代码

public StringBuffer appendTail(StringBuffer sb) {

sb.append(text, lastAppendPosition, getTextLength());

return sb;

}

友情链接

https://www.cnblogs.com/huhx/tag/java/default.html

java replaceall函数_java基础—-String中replace和replaceAll方法相关推荐

  1. python中的replaceall_java基础 — String中replace和replaceAll方法

    replace各个方法的定义 一.replaceFirst方法 public String replaceFirst(String regex, String replacement) { retur ...

  2. java虚拟函数_java实现多态中的虚函数相关概念

    本文转载自参考博客 1. Java虚函数 虚函数的存在是为了多态. C++中普通成员函数加上virtual关键字就成为虚函数 Java中其实没有虚函数的概念,它的普通函数就相当于C++的虚函数,动态绑 ...

  3. java获取数字_java 获取String中的数字

    随便给你一个含有数字的字符串,比如: String s="eert343dfg56756dtry66fggg89dfgf"; 那我们如何把其中的数字提取出来呢?大致有以下几种方法, ...

  4. java 检测表情符号_java – 检测String中的字符是否是表情符号(使用Android)

    我实际上能够使用链接的iOS代码来创建以下功能.我没有意识到包含例如单个表情符号的字符串将具有2的长度.因此,您可以检查字符是否实际上是代理. 我不完全确定如何从iOS代码中处理if(substrin ...

  5. java excel 透视_Java在Excel中创建透视表方法解析

    本文内容介绍通过Java程序在Excel表格中根据数据来创建透视表. 环境准备 需要使用Excel类库工具-Free Spire.XLS for Java,这里使用的是免费版,可通过官网下载Jar包并 ...

  6. C++string中replace()替换函数

    C++string中replace()函数 basic_string& replace(size_type pos, size_type n, const basic_string& ...

  7. android String的replace和replaceAll的使用

    今天,讲讲字符串中的替换字符的代码的使用. 一.replace的使用 声明 以下是Java.lang.String.replace()方法的声明 public String replace(char ...

  8. Java中删除String中空格的多种方法

    在Java中,要实现删除String中的空格的方法有很多种,具体有: 1. trim() 删除字符串开头和结尾的空格,Java 1版本引入的接口,这是开发人员最常用的方法.trim移除的空白字符指的是 ...

  9. java replaceall函数_JAVA中string.replace和string.replaceAll的区别及用法

    展开全部 JAVA中string.replace()和string.replaceAll()的区别及用法乍一看,字面上理解好像replace只替换第一个出现的字符(受javascript的影响),32 ...

最新文章

  1. python散点图拟合曲线-python – 将曲线拟合到散点图的边界
  2. 多线程必须用到的线程池(什么时候用多线程)
  3. DispacherServlet默认加载配置文件名
  4. Maven基础了解及配置信息
  5. springmvc怎么设置更改了界面不用重启_Microsoft Visual Studio 2019 更改语言包
  6. [jQuery] jQuery UI怎样自定义组件?
  7. 清华2018顶级数据团队建设全景报告发布:Python最受欢迎,C语言含金量最高
  8. c语言饭卡服务程序设计思路,C语言设计—饭卡管理程序.doc
  9. java 挥发注解_C中的挥发性预选赛
  10. Storm概念学习系列之并行度与如何提高storm的并行度
  11. 游戏开发之拷贝构造函数,即复制构造函数(explicit关键字及拷贝构造函数的调用时机)(C++基础)
  12. 基于netty构建http服务器
  13. 苹果说全是假的,市面不存在原装贴膜
  14. 有域名有服务器没有网站可以备案吗,只有域名没有服务器可以备案吗
  15. 计算机与经济学:天造地设的一对
  16. select 函数使用方法
  17. A类计算机机房温度变化,夏季什么样的温度才是机房适合的?
  18. 计算机组成原理17----CPU指令流水线
  19. java ee中ssh学后感_学习SSH,SSM框架有感
  20. 【CMS建站】写给大家看的网站制作教程03—零基础学网站制作的简单入门指南...

热门文章

  1. Hive两表count除法
  2. 斐波那契数列 递归与非递归算法实现
  3. 罗永浩宣布退网创业;谷歌研究员“走火入魔”事件曝光:认为AI已具备人格,被罚带薪休假;Wasmer 2.3 发布|极客头条
  4. 数据仓库 — 07_Kafka的安装与部署(Kafka命令行操作指令、Kafka集群群起脚本、压力测试、节点数量计算、hadoop_zookeeper_flume_kafka群起脚本、默认端口总结)
  5. 文献阅读:基于浅层残差网络的视线估计算法
  6. java微信支付验证签名_Java中的微信支付(3):API V3对微信服务器响应进行签名验证...
  7. 丁达尔效应产生的原因_为什么丁达尔效应产生的通路是乳白色的?
  8. 为什么要对基带信号进行脉冲成型【转载】
  9. ASP NET Web API 2框架揭秘
  10. Vegas如何制作相机模糊对焦效果?