深入理解java中==和equals的用法

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/zj15527620802/article/details/83309773

==号和equals()方法都是比较是否相等的方法,那它们有什么区别和联系呢? 
首先,==号在比较基本数据类型时比较的是值而用==号比较两个对象时比较的是两个对象的地址值

int x = 10;
int y = 10;
String str1 = new String("abc");
String str2 = new String("abc");
System.out.println(x == y); // 输出true ,str1这个引用变量存在虚拟机栈中,存放是"abc"这个对象的地址值
System.out.println(str1 == str2); // 输出false   ,str2str1这个引用变量存在虚拟机栈中,存放是两个不同的"abc"这个对象的地址值,两个对象的地址肯定不同,所以是false

那equals()方法呢?我们可以通过查看源码知道,equals()方法存在于Object类中,因为Object类是所有类的直接或间接父类,也就是说所有的类中的equals()方法都继承自Object类,而通过源码我们发现,Object类中equals()方法底层依赖的是==号,那么,在所有没有重写equals()方法的类中,调用equals()方法其实和使用==号的效果一样,也是比较的地址值,然而,Java提供的所有类中,绝大多数类都重写了equals()方法,重写后的equals()方法一般都是比较两个对象的值: 
 
这里我自己定义了一个Student类,没有重写equals()方法,最后的输出结果是:false

 
在我重写了equals()方法后,输出结果变成了true。

String 这个类重写了equals这个方法,所以比较的是两个对象的值。

现在一些基本的已经讲的差不多了,接下来我们回到第一个例子:

String str1 = new String("abc");//这个过程是运行时在堆内存中实时创建一个对象。
String str2 = new String("abc");//这个过程是运行时在堆内存中实时创建一个对象。
System.out.println(str1.equals(str2));//比较的是两个对象的值,是true
System.out.println(str1 == str2);//比较的是两个对象地址,是false

根据上面所讲,第一个是true,第二个是false,确实如此,那继续看下面的例子:

String s1 = "abc";
String s2 = "abc";
System.out.println(s1.equals(s2));
System.out.println(s1 == s2);

这次的结果和上一个的是一样的吗?答案是:true true 
为什么第二个会是true呢? 
这就涉及到了内存中的常量池,常量池属于方法区的一部分,当运行到s1创建对象时,在编译时期,查看如果常量池中没有这个对象,就在常量池中创建一个对象”abc”,第二次创建"abc",这个对象的时候,也是在编译时期,查看是否存在改对象,很明显,已经存在,这个时候,就会还回改对象的地址,所以s1,s2变量中存放是同一个对象的地址,很明显是想等等的。

那上一个例子中的

String str1 = new String("abc");
1
是怎么回事呢? 
这里其实创建了两次对象,依次是在常量池中创建了对象”abc”,一次是在堆内存中创建了对象str1,所以str1和str2的地址值不相等。

 

猜你喜欢

转载自blog.csdn.net/zj15527620802/article/details/83309773