1. String.intern()
看《深入理解Java虚拟机》提到运行时常量池(Runtime Constant Pool),运行时常量池是方法区的一部分,是存储什么的呢?
Class文件中除了有类的版本、字段,方法、接口等信息,还有常量池信息-Class文件常量池,存放了编辑期生成的各种字面量和符号引用,那么这些东西在类加载后就会被存放到方法区的运行时常量池中。
运行时常量池有个特性是具备动态性,java语言并不要求常量一定在编译期间产生,也就是除了上面说的编译期间产生的Class文件中的常量,运行期间也可能有新的常量产生并放入常量池,比较多的实际例子就是String类的intern()方法。
关于intern()的解释,《Java编程思想》里的解释是:为每一个字符序列生成且仅生成一个String引用,比较笼统,这里转载一篇比较清楚的blog:几张图轻松理解String.intern()。
2. String与StringBuilder
-
String对象不可变
String对象是不可变的,也即是任何看起来会修改String值的方法,实际都是创建了一个全新的对象,该对象包含了更改后的字符串内容,最初的对象不会受到改变。
-
重载“+”与StringBuilder
用于String的“+”和“+=”是java中仅有的两个重载过的操作符,因为java不允许程序员重载任何操作符。那么重载的“+”是做了什么优化么?
String mango = "mango";
String s = "abc" + mango + "def" + 47;
按照String的不变性,直观上上述代码在s的生成过程中每次+操作都会产生临时的一个String对象来存储中间结果,将会产生很多需要垃圾回收的中间对象,性能就不能保证了。然而实际上编辑器会自己优化这个操作,它会创建一个StringBuilder对象,每次+会调用append()方法,总计共四次,最后调用toString()生成最终结果存入s。
但是不要以为任何情况编辑器都会做出最好的优化,如果你的+或者+=是在循环中使用,那么StringBuilder的创建也是会在循环体里面,具体的例子可参看《Java编程思想》Ch13字符串。
3. StringBuilder
StringBuilder是Java SE5引入的,在这之前Java用的是StringBuffer,后者是线程安全的,因此开销会大一些。