我们知道在java中字符串拼接是非常常用的一个操作。有的用加号(+),有的用StringBuilder/StringBuffer.
经常有人问这些使用方式上有什么不同。于是就有了非常经典的答案,用加号链接字符串会在堆里创建额外的String对象(别跟我说JVM运行时常量池的预备值,看好了这里不讨论这个),从而影响性能,占用额外JVM堆内存空间,因此后一种用法比较好。然后进一步还会有人说StringBuffer比StringBuilder要重,因为StringBuffer是线程安全的。
今天在这里给大家澄清一下,如果你的JDK是1.5以后的话,那么上面的三种用法(+/StringBuilder/StringBuffer),没有差别。
原因是:JAVAC编译器(看好了,不是即时编译器JIT)会对上面风格的代码进行优化。
JDK1.5的的情况,如果JAVAC看到加号(+)链接字符串的代码,在编译成字节码时会自动替换成StringBuffer.append()方法。
这时又引入了新的问题,如果上面的String对象生命周期在一个方法内,而且没有发生引用逃逸,那么就在字节码中引入了多余的并发控制代码。
于是在即时编译期,JAVA即时编译器会进行检查,如果发现上面的代码没有进行同步的必要就会忽略掉上面代码的并发控制代码。这项技术叫做锁消除,JDK提高性能优化技术。
如果是JDK1.6的情况,遇到加号(+)链接的字符串会自动转化成StringBuilder.append(), 运行时上面的锁消除过程也省略了,效率更高。
知道以上这些,如果以后有人跟你说或让你讲这个问题时,你就可以给他做下update了。