使用字面量的方式创建字符串:
String s1 = "AB";
// "AB"是存储在方法区的字符串常量池当中
String s2 = "AB";
// 当执行String s1 = "AB"时,JVM首先会去字符串常量池中检查是否存在"AB"对象,如果不存在,则在字符串常量池中创建"AB"对象,并将"AB"对象的地址返回给s1;如果存在,则不创建任何对象,直接将字符串常量池中"AB"对象的地址返回给s1。所以s2直接拿到常量池中"AB"对象的地址,s2的"AB"不会新建,s1和s2都指向方法区的同一个对象"AB",内存地址相同。
// 用双等号(==)比较的是栈中变量中保存的内存地址,为true。
System.out.println(s1 == s2); // true
使用new对象的方式创建字符串:
String s1 = new String("AB");
String s2 = new String("AB");
System.out.println(s1== s2); // false
// 采用new关键字新建一个字符串对象时,JVM首先先在方法区的字符串常量池中查找有没有 "AB" 这个字符串对象,如果有,则不在字符串常量池中再去创建 "AB" 这个对象,如果没有,则创建出来。
// 随后会在堆中创建一个String类的对象,并指向字符串常量池的"AB"。
// 然后在将堆中的这个指向 "AB" 字符串的String类的对象自身的地址返回给引用s1。
// 每new一次都会在堆中创建一个新的String类的对象,所以每个引用保存的地址都不相同,因而双等号(==)比较的结果为false。
使用字面量相加的方式创建字符串:
String s1 = "abcdef";
String s2 = "abcdef" + "xy";
String s3 = "abcdef" + "xy";
System.out.println(s2==s3); // true
// 第一行代码会先在方法区的字符串常量池当中创建 "abcdef" 并把地址返回给s1保存,第二行代码中,因为 "abcdef" 在方法区的字符串常量池中已经存在,所以不会创建重复的 "abcdef" 对象,但 "xy" 这一字符串常量对象还未存在,则会在方法区的字符串常量池中创建一个 "xy" 字符串("xy"的地址不会返回给任何局部变量,因为没有设置变量去接收),随后会把他们拼接起来形成 "abcdefxy" 这一新的字符串常量并把新的字符串常量的地址返回给s2接收。
// 整个过程共创建了三个字符串对象。
// 凡是用双引号括起来的都是字符串常量区的一个字符串对象。
// s2和s3都是直接指向字符串常量池中的 "abcdefxy" 字符串,所以用双等号(==)比较的结果为true。
使用new关键字相加方式创建:
String s1 = new String("A") + new String("A");
String s2 = new String("A") + new String("A");
System.out.println(s1==s2); // false
// JVM首先先在方法区的字符串常量池中查找有没有 "A" 这个字符串对象,如果有,则不在字符串常量池中再去创建 "A" 这个对象,如果没有,则创建出来。(但不会返回地址给局部变量,因为没设局部变量去接收地址)
// 随后JVM会在字符串常量池中把 "A" 和 "A" 拼接起来形成新的字符串常量 "AA" (如果 "AA" 已存在则不在拼接),然后在堆中创建一个String类的对象并指向字符串常量池中的 "AA" ,最后返回堆中新创建的String类的对象的自身地址给引用s1。
// 每new一次都会在堆中创建一个新的String类的对象,所以每个引用保存的地址都不相同,因而双等号(==)比较的结果为false。
使用字面量和new关键字相加方式创建:
String s1 = "A" + new String("B");
String s2 = "A" + new String("B");
System.out.println(s1==s2); // false
// 与上面同理,s1和s2分别指向不同的堆对象,并且这两个堆对象都指向字符串常量池的同一个字符串常量 "AB" 。
使用equals方法可以避免这样的错误。
s1.equals(s2);