在Java编程中,字符串属于不可变对象,一旦创建就无法修改其内部字符内容,所有对字符串的修改操作都会生成新的字符串对象,频繁修改时会产生大量临时对象,带来额外的内存开销和性能损耗。如果需要高效修改字符串内容,将字符串转换为字符数组进行就地修改是常用的优化方案,而String.toCharArray()就是实现这个转换的核心方法。

String.toCharArray() 方法的基本原理
String类的toCharArray()方法会创建一个新的字符数组,数组的长度和原字符串的长度一致,并且会把字符串中每个索引位置的字符依次复制到数组的对应位置。由于返回的字符数组是新创建的可变对象,开发者可以直接修改数组中的元素,不需要额外创建字符串对象,这就是就地修改的核心逻辑。
该方法的源码实现逻辑比较简洁,核心操作就是分配对应长度的字符数组,然后调用底层的方法完成字符拷贝,以下是简化后的源码逻辑示意:
// String类toCharArray方法简化逻辑
public char[] toCharArray() {
// 创建和当前字符串长度一致的字符数组
char[] result = new char[value.length];
// 将字符串内部的字符数组拷贝到新数组中
System.arraycopy(value, 0, result, 0, value.length);
return result;
}
利用转换后的字符数组实现就地修改
得到字符数组之后,我们可以直接通过索引修改数组中的元素,修改完成后再将字符数组转换回字符串即可,整个过程只会创建一次字符数组和一次最终的字符串对象,相比直接拼接修改字符串的性能提升非常明显。
下面是将字符串转换为字符数组后修改内容的完整示例:
public class CharArrayModifyDemo {
public static void main(String[] args) {
// 原始字符串
String originalStr = "hello world";
System.out.println("原始字符串:" + originalStr);
// 调用toCharArray转换为字符数组
char[] charArray = originalStr.toCharArray();
// 就地修改字符数组内容,将索引6的字符从w改为W
charArray[6] = 'W';
// 将修改后的字符数组转回字符串
String modifiedStr = new String(charArray);
System.out.println("修改后的字符串:" + modifiedStr);
}
}
上述代码的执行结果是原始字符串为hello world,修改后的字符串为hello World,整个过程没有产生多余的字符串临时对象,实现了高效的就地修改。
不同修改方式的性能对比
为了直观展示利用toCharArray()进行就地修改的性能优势,我们可以对比三种常见的字符串修改方式的性能表现:直接字符串拼接修改、使用StringBuilder修改、使用toCharArray()转换后修改。以下是三种方式的实现和对比说明:
public class PerformanceCompare {
public static void main(String[] args) {
int loopCount = 100000;
String baseStr = "test";
// 方式1:直接字符串拼接修改
long start1 = System.currentTimeMillis();
String str1 = baseStr;
for (int i = 0; i < loopCount; i++) {
str1 = str1 + i;
}
long end1 = System.currentTimeMillis();
System.out.println("直接拼接耗时:" + (end1 - start1) + "ms");
// 方式2:使用StringBuilder修改
long start2 = System.currentTimeMillis();
StringBuilder sb = new StringBuilder(baseStr);
for (int i = 0; i < loopCount; i++) {
sb.append(i);
}
String str2 = sb.toString();
long end2 = System.currentTimeMillis();
System.out.println("StringBuilder耗时:" + (end2 - start2) + "ms");
// 方式3:使用toCharArray就地修改
long start3 = System.currentTimeMillis();
char[] charArray = baseStr.toCharArray();
// 这里模拟多次修改数组内容,实际场景根据需求调整
for (int i = 0; i < loopCount; i++) {
// 简单修改数组第一个字符,模拟修改逻辑
if (charArray.length > 0) {
charArray[0] = (char) ('a' + (i % 26));
}
}
String str3 = new String(charArray);
long end3 = System.currentTimeMillis();
System.out.println("toCharArray就地修改耗时:" + (end3 - start3) + "ms");
}
}
从性能表现来看,直接拼接修改的耗时最长,因为每次拼接都会创建新的字符串对象;StringBuilder的性能优于直接拼接,但在频繁修改单个字符的场景下,toCharArray()的就地修改方式性能表现更好,因为字符数组的索引修改是O(1)时间复杂度,不需要额外的扩容和拷贝操作。
使用注意事项
虽然toCharArray()的就地修改方式性能优秀,但使用时需要注意以下几点:
- toCharArray()返回的是新的字符数组,修改该数组不会影响原字符串的内容,因为字符串本身是不可变的。
- 如果转换后的字符数组长度不足以容纳修改后需要的内容,需要手动创建更长的数组并拷贝内容,避免数组索引越界。
- 如果是简单的字符串拼接场景,优先使用StringBuilder,toCharArray()更适合需要频繁修改单个字符、或者对字符位置有精确修改需求的场景。
- 修改完成后如果需要得到字符串对象,要使用
new String(charArray)的方式转换,不要直接使用字符数组的toString()方法,否则会得到对象的哈希值而不是字符串内容。
适用场景说明
这种转换后就地修改的方式适合以下场景:
- 需要对字符串中多个特定位置的字符进行替换修改,比如将字符串中的元音字母全部替换为指定字符。
- 需要按照自定义规则修改字符串中的字符,修改逻辑复杂,不适合用现有的字符串替换方法实现。
- 对性能要求较高的场景,比如循环中对大量字符串进行单字符修改,使用这种方式可以减少对象创建,提升运行效率。
下面是一个实际的应用示例,将字符串中的所有小写字母转换为大写字母,使用toCharArray()实现:
public class UpperCaseDemo {
public static String toUpperCaseByCharArray(String str) {
if (str == null || str.isEmpty()) {
return str;
}
// 转换为字符数组
char[] arr = str.toCharArray();
// 遍历数组就地修改
for (int i = 0; i < arr.length; i++) {
if (arr[i] >= 'a' && arr[i] <= 'z') {
arr[i] = (char) (arr[i] - 32);
}
}
// 转回字符串
return new String(arr);
}
public static void main(String[] args) {
String test = "hello Java String Demo";
System.out.println("转换结果:" + toUpperCaseByCharArray(test));
}
}
这个示例通过遍历字符数组,判断每个字符是否为小写字母,如果是就转换为对应的大写字母,整个过程高效且逻辑清晰,适合处理这类单字符批量修改的需求。
String_toCharArray字符数组就地修改字符串转换Java修改时间:2026-07-04 01:18:37