

新闻资讯
行业动态绝大多数场景优先用StringBuilder;线程安全时选StringBuffer;常量拼接由编译器自动优化。关键在“谁在改”和“谁可能同时改”:String不可变,StringBuilder可变非线程安全,StringBuffer线程安全但有锁开销。
绝大多数场景下,优先用 StringBuilder;需要线程安全时才考虑 StringBuffer;纯拼接常量或简单表达式,编译器会自动优化成 StringBuilder,不用手动干预。
关键区别不在“能不能改”,而在“谁在改”和“谁可能同时改”:
String 不可变,每次 + 或 substring() 都新建对象,高频拼接或循环中反复操作会触发大量 GCStringBuilder 可变、非线程安全,单线程下性能最优,内部用 char[](Java 9+ 是 byte[] + 编码标记)动态扩容StringBuffer 方法全加了 synchronized,多线程安全但锁开销明显,除非明确在共享可变字符串对象且无法重构为无状态,否则不推荐默认构造函数创建的 StringBuilder 初始容量是 16,一旦追加内容超过该长度,就会触发数组扩容——本质是新建更大数组、复制旧内容。高频小拼接(如日志组装、SQL 拼接)若没预估长度,可能反复扩容拖慢性能。
实操建议:
new StringBuilder(256)
String.join() 更简洁(Java 8+),它内部也用 StringBuilder,但做了容量预计算StringBuilder:把实例提到循环外复用,尤其在方法内部被多次调用时toString().length() 判断是否为空,用 length() == 0,前者多一次对象创建StringUtils 能省多少事原生 String API 对空值处理极其脆弱:str.trim() 在 str == null 时直接抛 NullPointerException。而 StringUtils 几乎所有方法都做了 null 安全封装,且语义更贴近日常需求。
典型替代场景:
StringUtils.isEmpty(str) 等价于 str == null || str.length() == 0,比 Objects.isNull(str) || str.isEmpty() 少写一半StringUtils.isBlank(str) 还会检查空白字符(如 "\t\n ")StringUtils.substring(str, 0, 10) 在 str == null 或越界时不报错,返回 null 或原串StringUtils.join(list, ", ") 自动跳过 null 元素,无需提前过滤注意:它不替代 StringBuilder,而是补足原生 API 的健壮性缺口。引入 
commons-lang3 依赖后,这些方法基本零学习成本。
String.indent() 和 transform() 值不值得用这两个方法属于“写起来爽、读起来清”的语法糖,但需留意兼容性和实际效果:
str.indent(4) 在每行前加 4 个空格,但会移除原字符串首尾空白行,且对首行缩进逻辑特殊(不缩进首行换行符后的部分)——容易和预期不符,调试时建议先打印结果str.transform(s -> s.toUpperCase()) 本质就是 s.toUpperCase().toString() 的包装,唯一优势是支持链式调用、延迟执行(配合函数式接口),但无性能增益StringBuilder,纯属 String 实例方法增强;若项目还停留在 Java 11 或更低,它们根本不可用真正需要关注的是 String 的底层存储变化:Java 9 开始用 byte[] + coder 字段节省内存(Latin-1 编码下空间减半),但这对开发者透明,只影响 GC 压力和堆占用。