创建String很特别:有两种String s = "abc"; String s = new String("abc");
字符串的存储也是很特别的,存储分两种:
第一种的创建方式是在编译的时候把“abc”存到字符串常量池(这个涉及到jvm的内存模型),可以看成一个特殊的堆内存;第二种创建方式会在运行期间生成一个String对象,这时候是把它存到堆内存,像其他普通对象的创建一样。
就是因为这两个区别,造成了字符串的==操作(比较的是地址)会产生奇怪的结果:
例子一:
String s0 = "world";
String s1 = "hello";
String s2 = "hello" + "world";
String s3 = "hello" + "world";
System.out.println(s2 == s3);//true 编译期间明确知道是常量,所有取值唯一。
String s4 = s1 + s0;
String s5 = s1 + s0;
System.out.println(s4 == s5);
//false 编译期间取得是栈引用,所以编译的时候不知道内容,所以新建到堆中,取值不一致
这个例子中:第一个输出结果为true应该是能懂的,因为hello + world的操作在编译的时候就已经知道生成“helloworld”这个字符串了,所以会把helloworld放到字符串常量池,s2 和 s3都是指向这个常量,所以比较的时候地址是一样的。
而对于下面的false,我当时想:s0和是s1都是在编译期间指向了常量池中的字符串对象,然而相加操作编译期间应该也是确定了的啊,为什么s4和s5是不一样的对象呢?后来仔细想了下:在编译期间能确定“hello”,“world”,“helloworld”这三个字符串常量,这是没问题的,但是s1和s0的指向内容在编译期间能确定吗?这个问题就是关键所在,对于变量的指向内容编译期间是不知道的,最简单的例子就是写了一个程序报空指针异常,正是因为编译期间不知道变量的指向,所以才会报空指针异常啊。理解了这个道理,上面的结果为什么为false就一目了然了,就是因为s0和s1的指向内容编译期间根本不知道(尽管我们很清楚得看出来它的指向内容),所以s4和s5的创建都是新建String对象,而且放在堆上不同的位置,所以用==结果为false。
另外一个就是intern方法,这个就是返回一个字符串在字符串常量池中的引用,如果这个字符串不在常量池中,那么就把这个字符串放到常量池中,再返回常量池中这个字符串的引用。这个就是动态扩充字符串常量池的一个例子,所以说常量池不仅仅是在初始化的时候才添加。
package cn.wzy.string;
/**
* @author wzy 不短不长八字刚好.
* @since 2018/9/1 9:47
*/
public class StringTest {
public static void main(String[] args) {
String s0 = "world";
String s1 = "hello";
String s2 = "hello" + "world";
String s3 = "hello" + "world";
System.out.println(s2 == s3);//true 编译期间明确知道是常量,所有取值唯一。
String s4 = s1 + s0;
String s5 = s1 + s0;
System.out.println(s4 == s5);
//false 编译期间取得是栈引用,所以编译的时候不知道内容,所以新建到堆中,取值不一致
System.out.println("========================");
String s6 = new String("asdf");
System.out.println(s6 = s6.intern());
//运行期间扩充字符串常量池
System.out.println(s6 == "asdf");
}
}