判断字符串为空或空白应先判null再trim().isEmpty(),或用StringUtils.isBlank()、JDK11+的isBlank();substring endIndex不包含且需满足索引范围;replace处理字面量,replaceAll处理正则;循环拼接必须用StringBuilder。

如何判断字符串是否为空或只含空白字符
别直接用 str == null || str.length() == 0,这漏掉空格字符串;也别迷信 str.isEmpty(),它对 null 会抛 NullPointerException。
- 安全写法是先判
null,再用str.trim().isEmpty(),或直接用 Apache Commons 的StringUtils.isBlank(str) - JDK 11+ 可用
str == null || str.isBlank()(isBlank()是String自带方法,会检查空、空格、制表符等 Unicode 空白) -
" \t\n".isEmpty()返回false,但" \t\n".isBlank()返回true
substring() 的索引边界到底怎么算
substring(int beginIndex, int endIndex) 的 endIndex 是**不包含**的,且两个参数都必须满足 0 ≤ beginIndex ≤ endIndex ≤ str.length()。越界就抛 StringIndexOutOfBoundsException。
-
"hello".substring(1, 4)→"ell"(下标 1 到 3) -
"hello".substring(3, 3)→""(合法,返回空串) -
"hello".substring(5)→""(单参版本从指定位置截到末尾) - 常见错误:把
endIndex当成“长度”,写成str.substring(i, i+3)却没检查i+3 ≤ str.length()
replace() 和 replaceAll() 到底该用哪个
看名字容易误以为 replaceAll() 更“强”,其实它们根本不是同一类操作:replace() 处理字面量,replaceAll() 处理正则表达式。
-
"a.b.c".replace(".", "_")→"a_b_c"(正确替换所有点号) -
"a.b.c".replaceAll(".", "_")→"_____"(错误!.在正则里匹配任意字符) - 要正则替换,得转义:
"a.b.c".replaceAll("\\.", "_") - 如果只是字符/字符串替换,无条件优先选
replace()—— 更快、更安全、语义清晰
concat()、+ 和 StringBuilder.append() 性能差异在哪
编译器会对字符串字面量拼接做优化,但运行时拼接行为完全不同:
立即学习“Java免费学习笔记(深入)”;
-
"a" + "b"编译期就合并为"ab";s1 + s2(变量)会被编译成new StringBuilder().append(s1).append(s2).toString() -
str.concat("x")底层是新建 char 数组复制,比+稍轻量,但仅适用于单次追加 - 循环内拼接必须用
StringBuilder,否则每次+都新建对象,O(n²) 时间复杂度 - JDK 9+ 对
StringConcatFactory做了优化,但逻辑复杂,别依赖——明确场景用StringBuilder最稳
String result = "";
for (String s : list) {
result += s; // ❌ 千万别这么写
}
// ✅ 正确写法:
StringBuilder sb = new StringBuilder();
for (String s : list) {
sb.append(s);
}
String result = sb.toString();
字符串不可变性是所有行为的基础,任何“修改”都是创建新对象。很多 bug 和性能问题,根源都在忘了这一点。










