java中字符串的初始化过程以及String Stringbuilder StringBuffer 的区别:
在Java的JVM中,有一个字符串常量池的概念,在jdk 1.7之后,字符串常量池被划分到java的Heap中,Java的八种基本数据类型中,除了float 和 Double ,其他都实现了常量池技术。
(字符串常量池底层是用HashTable实现的,以 key---value 的形式存储数据)
1. String s1 = "hello world"; String s2 = new String("hello world"); 有什么区别呢?
System.out.println(s1 == s2); 打印的结果是True 还是 False ?
结论:使用双引号的形式直接声明字符串,在JVM 中的操作过程是:(1)JVM首先会到字符串常量池中查找有没有“hello world”字 符串,如果有,那么直接引用常量池中的。(2)如果没有,那么JVM首先会在堆内存(Heap)中new一个字符串对象,然后在常量池中注册一个该字符串。
使用new关键字创建一个字符串,在JVM 中的操作过程是:JVM不会查询字符串常量池,会直接在Heap中创建一个新的对 象。
所以上面打印的结果为False。
2. String s1 = new String("hello") + new String("world");
s1.inter();
String s2 = "hello world";
System.out.println(s1 == s2);
该结果输出为True,当s1调用inter();时,JVM首先会去常量池中查找是否有该字符串的引用,如果有,直接返回该引用,
如果没有,JVM会将s1字符串的引用注册到常量池中,当s2创建时,会从常量池中找得到,所以s2与s1 相等。
3. String s1 = "hello"; String s2 = "world";
String s3 = s1 + s2; String s4 = "hello world";
System.out.println(s3 == s4);
解决这个问题的关键是要判断s3中的s1+s2是以双引号的形式创建,还是以new的形式创建。使用 javap -c 反编译代码,可以看出,两个字符串相加,底层使用的是StringBuilder.append(); ,然后会以StringBuilder.toString()的形式输出,在toString();中,使用了 new String对象的形式返回一个字符串,所以s3是通过new 的形式来创建的,JVM不会将 s3 写入常量池中,所以 s4 也就不会从常量池中拿到 "hello world"字符串,所以输出的结果为false。
4.String , StringBuffer , StringBuilder 的底层实现:
(1) String ,StringBuffer , StringBuilder 底层都是采用 char 类型的数组实现。
(2) String 为不可变类型的字符串,在String 类中,所有返回值类型为String类型的方法中,都会 new 一个新的字符串。
StringBuffer 和 StringBuilder 会使用append();在现有的字符串基础上拼接得到。但是StringBuffer类中的方法,添加了
Synchronized关键字,它是线程安全的,所以效率会比StringBuilder低。
(3) 使用String 类型进行字符串的拼接时,每一次拼接都会生成一个StringBuilder对象,调用append(); 然后在调用 toString(),
通过new 一个新的字符串的形式来返回。在进行多次拼接时,效率会很低。