
本文介绍一种高效、可扩展的字符串处理方法:遍历字符串时动态统计指定字符的累计出现次数,并将每次匹配位置的该字符原地替换为当前序号(如第1次出现替换成'1',第2次替换成'2',以此类推),支持任意长度数字的无缝插入。
在实际开发中,常需对字符串中特定字符进行“序号化标记”——即把第 n 次出现的某个字符(如 'l')替换为数字 n(如 '1', '2', '10', '100')。例如:
- 输入 "helololol",替换 'l' → "he1o2o3o4"
- 输入 "hellollololollol",替换 'l' → "he1o2lo3o4lo5o6llo7o8lo9l10ol11o12"
⚠️ 注意:不能直接用 char[] 原地修改(如 arr[2] = '1'),因为数字可能不止一位(如 10 占 2 字符),导致数组越界或覆盖后续内容;也不能简单用 String.substring() 拼接而不调整索引——随着字符串增长,原下标会失效。
✅ 推荐方案:使用 StringBuilder 动态构建(安全、高效、可扩展)
public static String replaceWithCount(String str, char target) {
StringBuilder sb = new StringBuilder();
int count = 1;
for (int i = 0; i < str.length(); i++) {
char c = str.charAt(i);
if (c == target) {
sb.append(count++);
} else {
sb.append(c);
}
}
return sb.toString();
}
// 使用示例
public static void main(String[] args) {
System.out.println(replaceWithCount("hello", 'l')); // he12o
System.out.println(replaceWithCount("helololol", 'l')); // he1o2o3o4
System.out.println(replaceWithCount("hellollololollol", 'l')); // he1o2lo3o4lo5o6llo7o8lo9l10ol11o12
}? 关键设计说明
- 无索引漂移问题:StringBuilder 顺序追加,完全规避了字符串拼接导致的下标错位风险;
- 天然支持大数字:count 可无限递增(int 范围内),append(count) 自动转为多位数字字符串(如 100 → "100");
- 时间复杂度 O(n):单次遍历,无嵌套操作;
- 空间友好:仅额外使用 StringBuilder 缓冲区,无频繁字符串对象创建(避免 substring + concat 的内存开销)。
⚠️ 补充注意事项
- 若需区分大小写,确保 target 字符与源字符串严格匹配(Java 默认区分);
- 若需批量替换多个字符,可扩展为 Map
维护各字符独立计数器; - 如需保留原始字符格式(如 Unicode 表情、中文),本方案同样适用,因 charAt() 和 StringBuilder 全面支持 UTF-16;
✅ 总结
用 StringBuilder 逐字符扫描并条件追加,是实现“按序替换为计数编号”的最优实践:简洁、健壮、可读性强,且天然适配任意规模数据。避免手动下标计算和字符串拼接陷阱,让逻辑回归本质——一次遍历,一次构建。










