charAt()越界会抛StringIndexOutOfBoundsException;String底层是char[]但不暴露,取字符必须用charAt(),调用前需检查index>=0&&index

用 charAt() 取字符时索引越界会直接抛异常
Java 中字符串不可变,String 本质是 char[] 封装,但不暴露数组本身。想取单个字符必须用 charAt(int index) —— 它不做边界静默处理,越界立即抛 StringIndexOutOfBoundsException。
- 检查长度:调用前务必确认
index >= 0 && index - 循环遍历时推荐
for (int i = 0; i ,而非for (int i = 0; i (常见 off-by-one) - 空字符串
""的length()是 0,对它调用charAt(0)必然崩溃
把 char 拼成字符串别用 + 做循环累加
char 是基本类型,和字符串拼接时会自动装箱为 Character,再通过 StringBuilder 转换。但循环中反复用 += 会产生大量中间 String 对象,性能极差。
- 正确做法:初始化
StringBuilder,用append(char)累加,最后调toString() - 如果只是少量拼接(比如 2~3 个字符),
"" + a + b + c可读性尚可,编译器会优化成StringBuilder - 避免写
String s = ""; for (char c : arr) s += c;—— 这是典型 O(n²) 行为
StringBuilder sb = new StringBuilder();
for (char c : charArray) {
sb.append(c);
}
String result = sb.toString();
判断字符串是否只含某个字符,别用 contains(String) 或正则
想确认一个 String 是否“全部由同一个 char 组成”,比如 "aaaa" 或 " ",用 contains() 或 .matches("a+") 都是错的路径 —— 前者只查存在性,后者要编译正则、开销大且易写错模式。
- 最简方案:先取首字符
str.charAt(0),再用str.chars().allMatch(c -> c == targetChar) - 兼容 Java 8 以下:用传统循环,
for (int i = 1; i - 注意空字符串:需单独判断
str.isEmpty(),否则charAt(0)报错
String.valueOf(char) 和 new String(char[]) 的区别很实在
两者都能从字符生成字符串,但语义与开销不同:
立即学习“Java免费学习笔记(深入)”;
-
String.valueOf(c)是静态工厂方法,对单个char直接返回缓存的字符串对象(内部用Character.toString(c)),无对象分配开销 -
new String(new char[]{c})强制新建String实例,即使内容相同也 ≠ 已有字符串(==为 false),且多一次数组创建 - 批量转字符串时:
String.valueOf(charArray)比new String(charArray)更推荐,前者可复用底层数组(JDK 9+ 使用byte[]存储,但逻辑一致)
// 推荐
String s1 = String.valueOf('x');
String s2 = String.valueOf(charArray);
// 不必要
String s3 = new String(new char[]{'x'});
String s4 = new String(charArray);
字符操作看似简单,但 charAt() 的异常行为、字符串拼接的隐式开销、以及工厂方法与构造器的语义差异,都是线上容易突然冒出来的点。尤其在处理用户输入或解析协议字段时,空串、单字符、超长字符串这三类边界组合起来,最容易漏判。










