目录
典型回答
String的"+"是如何实现的
StringBuffer和StringBuilder
不要在for循环中使用+拼接字符串
-
典型回答
- 本质上都是char[]字符数组的实现
- 在Java9之后,String类的实现改用byte数组存储字符串
- 使用final关键字修饰字符数组来保存字符串,String是不可变的
- 没有使用final关键字修饰,StringBuilder和StringBuffer是可变的
- String中的对象是不可变的,也就可以理解为常量,线程安全
- StringBuffer对方法加了同步锁或者对调用的方法加了同步锁,是线程安全的
- StringBuilder并没有对方法进行加同步锁,是非线程安全的
- 每次要对String类型进行改变时,都会生成一个新的String对象,然后将指针指向新的String对象
- StringBuffer每次都会对StringBuffer对象本身进行操作,而不是生成新的对象并改变对象引用
- 相同情况下使用StringBuilder相比使用StringBuffer仅能获得10%-15%左右的性能提升,但却要冒多线程不安全的风险
- 使用场景:
- 操作少量的数据用String,但是常改动内容且操作数据多情况下最好不要用String,因为每次生成中间对象性能会降低
- 单线程下操作大量的字符串用StringBuilder,虽然线程不安全但是不影响
- 多线程下操作大量字符串,且需要保证线程安全,则用StringBuffer
-
String的"+"是如何实现的
- 使用+拼接字符串,其实只是Java提供的一个语法糖,那么就来解一解这个语法糖,看看他的内部原理到底是如何实现的
- 还是这样一段代码
- 把他生成的字节码进行反编译,看看结果
- 反编译后的内容如下,反编译工具为jad
- 通过查看反编译以后的代码,可以发现,原来字符串常量在拼接过程中,是将String转成了StringBuilder后,使用其append方法进行处理的
- 那么也就是说,Java中的+对字符串的拼接,其实现原理是使用StringBuilder.append
-
StringBuffer和StringBuilder
- 与 String 不同,字符数组中不一定所有位置都已经被使用,它有一个实例变量,表示数组中已经使用的字符个数,定义如下:
- 其append源码如下:
- 该类继承了`AbstractStringBuilder`类,看下其`append`方法:
- append会直接拷贝字符到内部的字符数组中,如果字符数组长度不够,会进行扩展
-
不要在for循环中使用+拼接字符串
- 前面分析过,其实使用`+`拼接字符串的实现原理也是使用的`StringBuilder`,那为什么不建议在for循环中使用呢?
- 反编译后代码如下:
- 可以看到,反编译后的代码,在 for 循环中,每次都是 new 了一个 StringBuilder ,然后再把 String 转成 StringBuilder ,再进行 append
- 而频繁的新建对象当然要耗费很多时间了,不仅仅会耗费时间,频繁的创建对象,还会造成内存资源的浪费
- 所以,阿里巴巴Java开发手册建议:循环体内,字符串的连接方式,使用 StringBuilder 的 append 方法进行扩展;而不要使用 +