
本文介绍如何遍历字符串,对每个匹配的指定字符动态替换为“其当前出现序号”,实现如 `"helololol"` → `"he1o2o3o4"` 的转换,并提供可扩展、健壮的 java 实现方案。
在实际开发中,有时需要将字符串中某类字符(如 'l' 或 'o')按其从左到右第几次出现进行编号替换——即首次出现替换为 '1',第二次为 '2',依此类推。该需求看似简单,但需注意:字符串不可变性和索引偏移问题(替换后字符串长度变化,后续字符位置前移),若直接用 substring 拼接处理不当,极易引发 IndexOutOfBoundsException 或逻辑错位。
以下是一个安全、清晰且可扩展的解决方案:
✅ 推荐实现:使用 StringBuilder + 单次遍历
public static String replaceCharWithCount(String str, char target) {
if (str == null) return null;
StringBuilder sb = new StringBuilder();
int count = 0;
for (int i = 0; i < str.length(); i++) {
char c = str.charAt(i);
if (c == target) {
count++;
sb.append(count); // 直接追加数字(自动支持多位数,如 10、100)
} else {
sb.append(c);
}
}
return sb.toString();
}✅ 使用示例:
public static void main(String[] args) {
System.out.println(replaceCharWithCount("helololol", 'l')); // 输出: he1o2o3o4
System.out.println(replaceCharWithCount("helololol", 'o')); // 输出: hel1l2l3l
System.out.println(replaceCharWithCount("hellollololollollol", 'l')); // 自动支持 >9 次:he12o34o56o78o910...
}⚠️ 关键注意事项
- 避免字符串拼接陷阱:原答案中 str = str.substring(0,i) + num++ + str.substring(i+1) 在循环中反复创建新字符串,时间复杂度为 O(n²),且当 num ≥ 10 时,substring(i+1) 的起始索引会因前面插入了两位数而失效(例如 '10' 占 2 位,但代码仍只跳过 1 位),导致覆盖错误或越界。
- StringBuilder 是最优解:它内部维护可变字符数组,append() 均摊时间复杂度为 O(1),天然规避索引漂移问题,且自动处理任意位数整数(1、12、123 等均正确转为字符串拼接)。
- 无需预统计或反向遍历:单次正向扫描即可完成,空间复杂度 O(n),简洁高效。
? 扩展建议(进阶场景)
- 若需区分大小写或支持正则匹配(如所有数字、空格等),可将 char target 替换为 Predicate
; - 若需保留原始字符并附加计数(如 'l' → "l1"),仅需修改 sb.append(count) 为 sb.append(c).append(count);
- 如需格式化计数(如补零:"01", "02"),可用 String.format("%02d", count)。
总之,面对“边遍历边修改”的字符串处理任务,优先选择 StringBuilder 构建结果,而非在原字符串上做破坏性拼接——这是保证逻辑正确性与性能稳定性的关键实践。










