Answers:
String target = "FOOBar";
target = target.replaceAll("(?i)foo", "");
System.out.println(target);
输出:
Bar
值得一提的是,replaceAll
将第一个参数视为正则表达式模式,这可能会导致意外结果。要解决此问题,请Pattern.quote
按照注释中的建议使用。
target.replaceAll("(?i)"+Pattern.quote("foo"), "");
也许不如其他方法那么优雅,但是它非常扎实且易于遵循,尤其是。对于刚接触Java的人。让我了解String类的一件事是:它已经存在很长时间了,虽然它支持使用regexp进行全局替换和使用Strings(通过CharSequences)进行全局替换,但最后一个没有简单的布尔参数:'isCaseInsensitive'。确实,您曾想过,只需添加一个小开关,就可以避免它的缺失给初学者带来的所有麻烦。现在在JDK 7上,String 仍然不支持这一功能!
好吧,我会停止抓紧。对于特别喜欢Java的每个人,这里都是您可以剪切并粘贴的deus ex machina。就像我说的那样,它不那么优雅,不会赢得任何出色的编码奖,但是它有效并且可靠。任何意见,请随时贡献。(是的,我知道,StringBuffer可能是管理两个字符串突变行的更好选择,但交换技术很容易。)
public String replaceAll(String findtxt, String replacetxt, String str,
boolean isCaseInsensitive) {
if (str == null) {
return null;
}
if (findtxt == null || findtxt.length() == 0) {
return str;
}
if (findtxt.length() > str.length()) {
return str;
}
int counter = 0;
String thesubstr = "";
while ((counter < str.length())
&& (str.substring(counter).length() >= findtxt.length())) {
thesubstr = str.substring(counter, counter + findtxt.length());
if (isCaseInsensitive) {
if (thesubstr.equalsIgnoreCase(findtxt)) {
str = str.substring(0, counter) + replacetxt
+ str.substring(counter + findtxt.length());
// Failing to increment counter by replacetxt.length() leaves you open
// to an infinite-replacement loop scenario: Go to replace "a" with "aa" but
// increment counter by only 1 and you'll be replacing 'a's forever.
counter += replacetxt.length();
} else {
counter++; // No match so move on to the next character from
// which to check for a findtxt string match.
}
} else {
if (thesubstr.equals(findtxt)) {
str = str.substring(0, counter) + replacetxt
+ str.substring(counter + findtxt.length());
counter += replacetxt.length();
} else {
counter++;
}
}
}
return str;
}
由于保留了一些字符,因此正则表达式的管理非常复杂:例如,"foo.bar".replaceAll(".")
生成一个空字符串,因为点表示“任何内容”。如果要替换,则仅将点指示为参数"\\."
。
一个更简单的解决方案是使用StringBuilder对象搜索和替换文本。它需要两个:一个包含小写版本的文本,第二个包含原始版本。搜索是对小写内容进行的,并且检测到的索引也将替换原始文本。
public class LowerCaseReplace
{
public static String replace(String source, String target, String replacement)
{
StringBuilder sbSource = new StringBuilder(source);
StringBuilder sbSourceLower = new StringBuilder(source.toLowerCase());
String searchString = target.toLowerCase();
int idx = 0;
while((idx = sbSourceLower.indexOf(searchString, idx)) != -1) {
sbSource.replace(idx, idx + searchString.length(), replacement);
sbSourceLower.replace(idx, idx + searchString.length(), replacement);
idx+= replacement.length();
}
sbSourceLower.setLength(0);
sbSourceLower.trimToSize();
sbSourceLower = null;
return sbSource.toString();
}
public static void main(String[] args)
{
System.out.println(replace("xXXxyyyXxxuuuuoooo", "xx", "**"));
System.out.println(replace("FOoBaR", "bar", "*"));
}
}
我喜欢smas与正则表达式一起使用的答案replaceAll
。如果您打算多次进行相同的替换,则一次预编译正则表达式是有意义的:
import java.util.regex.Pattern;
public class Test {
private static final Pattern fooPattern = Pattern.compile("(?i)foo");
private static removeFoo(s){
if (s != null) s = fooPattern.matcher(s).replaceAll("");
return s;
}
public static void main(String[] args) {
System.out.println(removeFoo("FOOBar"));
}
}
不用第三方库就可以使其变得简单:
final String source = "FooBar";
final String target = "Foo";
final String replacement = "";
final String result = Pattern.compile(target, Pattern.LITERAL | Pattern.CASE_INSENSITIVE | Pattern.UNICODE_CASE).matcher(source)
.replaceAll(Matcher.quoteReplacement(replacement));