String的比较,看了好多相关的面试题和相关的博客,讲解大都不是很透彻、全面,我就自己总结了一下。希望大家对String的比较面试题,有较为全面的理解了。
问题的核心:==是判断地址值是否相等的,主要就是常量池和堆的地址值对比!
String str1 = new String("abc");
Stirng str2 = "abc";
String是一个非可变类(immutable),虽然两个语句都是返回一个String对象的引用,但是jvm对两者的处理方式是不一样的。
- 对于第一种,jvm第一个创建的对象就是在常量池里的“abc”。然后在堆中创建一个String对象,然后将该堆中对象的引用返回给用户。创建了两个对象,一个在常量池,一个在堆中。
- 对于第二种,jvm首先会在内部维护的strings pool中,通过String的 equels 方法查找是对象池中是否存放有该String对象,如果有,则返回已有的String对象给用户;如果对象池中没有该String对象,常量池创建一个“abc”对象,并把常量池的引用返回。只创建了一个常量池中的对象。
特殊场景1:
String str1 = "hello quanjizhu";
String str2 ="hello" +"quanjizhu";
在编译的时候会优化成String str2 = "hello quanjizhu";所以str1和str2指向的是同一内存地址。
特殊场景2:
String var = “quanjizhu“;
String str4 = “hello “ + var;
System.out.println(str1 == str4);
输出结果是false,证明了String str4 = “hello “+var;在内存堆中会重新分配空间(相当于使用构造方法创建新对象),而不是让str4指向var的地址。这个拼接的原理是由StringBuilder或者StringBuffer类里面的append方法实现拼接,然后调用toString()把拼接的对象转换成字符串对象,最后把得到字符串对象的地址赋值给变量,没有在常量池中创建对象。
特殊场景3:
String s1 = "abc";
String s3 = new String("abc");
s3 = s3.intern();
System.out.println(s1= =s3)
输出结果是true;intern运行过程是这样的:首先查看strings pool有没“abc”对象的引用,有则直接返回该常量池对象的引用;没有,则在堆中新建一个对象,然后将新对象的引用加入至strings pool中,并返回该常量池的对象(即堆中的地址)。
此处因为常量池已经存在了"abc",所以intern返回的是常量池的地址,所以s1=s3。
如果是jdk1.7以上,那么就在常量池保存指向堆中"s3"的地址,即保存堆中"s3"的引用。
案例1:
String s3 = new String("hello");//常量池、堆中都会新增
s3.intern();//常量池中已有,不新增
String s1 = "hello";
System.out.println(s1 == s3);//false
案例2:
String str2 = new String("str")+new String("01");//在堆中新建对象,没有在常量池中创建对象。
str2.intern();//保存堆中的引用
String str1 = "str01";//常量池已有,直接返回堆中的引用
System.out.println(str2==str1);//true
总结:
String str2 = new String("str");//在堆和常量池中都会创建,地址不同,返回的是堆中的地址
String str1 = "str01";//只在常量池中创建,直接返回常量池中的地址
new String("str") + new String("01"); //相当于调用StringBuilder或者StringBuffer类里面的append方法实现拼接,然后调用toString()把拼接的对象转换成字符串对象,最后把得到字符串对象的地址赋值给变量,保存在堆中的地址,没有在常量池中创建对象"str01",但创建了"str"和"01"。
intern(); //首先查看常量池中有没“abc”这个字符串,有则直接返回该常量池对象的引用;没有,则在常量池中生成一个对堆中对象的引用,并返回该常量池的对象(即堆中的地址)。