在Java里如何处理字符串与字符_JavaString与char操作说明

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[]) 的区别很实在

两者都能从字符生成字符串,但语义与开销不同:

  • 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() 的异常行为、字符串拼接的隐式开销、以及工厂方法与构造器的语义差异,都是线上容易突然冒出来的点。尤其在处理用户输入或解析协议字段时,空串、单字符、超长字符串这三类边界组合起来,最容易漏判。