栈区存引用和基本类型,不能存对象,而堆区存对象。==是比较地址,equals()比较对象内容。
在hotspot虚拟机中,jdk1.8之后,String常量池是在堆中的,单纯的String str="aaa",是从虚拟机栈中的变量直接指向String常量池的中的对象,但是如果采用new关键字,那么他会首先在堆中开辟一片空间,存储该字符串在常量池中的地址,new两次就开辟两个,虽然他们会指向同一个String常量池,但是堆中地址不一样,也就是new之前的变量指向不一样,所以他们不相等
下面通过一个例子来讲述两者的区别
public class Test3 {
public static void main(String[] args) {
String a=new String("myString");
String b="myString";
String t1="my";
String t2="String";
String t=t1+t2;
String c="my"+"String";
String d=c;
String f=new String("myString");
System.out.println(a==b);
System.out.println(a==c);
System.out.println(b==c);
System.out.println(b==d);
System.out.println(a==f);
System.out.println(b==t);
}
}
看到以上代码,你会觉得输出什么?
true true true true true true或者false true false true true false等等猜想,下面就让我跟大家讲解一下过程吧。
String a=new String("myString")的实现过程:直接在堆中创建对象,如果后来又有String f=new String("myString"),f不会指向之前的对象,而是重新创建一个对象并指向它,所以如果此时进行的是a==f,那么返回值是false,因为两个对象的地址不同,如果是a.equals(f),返回的结果是true,因为内容相同。
String b="myString"的实现过程:首先在栈区创建b引用,然后在常量池中寻找内容为"myString"的对象,如果没有,则创建一个,然后b指向Stirng池中的对象,如果有,则直接将b指向"myString";如果后来又定义了b1="myString",则指向将b1引用指向String池中已经存在的"myString",不在新建对象。hotspot中编译"my"+"String"将直接变成"myString";t1+t2则不会优化,因为不知道在之前的操作中t1,t2是否发生改变;而针对t1+t2则是用语法糖,新建一个StringBuilder来处理。
自己可以多打几个例子。。。