Java面试分享——“==”与equals的区别,源码解析

前言:面试被问到答了个七七八八,但是总是不太熟练于是写下了这一篇博客,期待自己将他牢记于心。

在学习Java前期你是否碰到过这样的疑问?

String str1=new String("aaa");
String str2=new String("aaa");

System.out.println("第一次比较:"+(str1==str2));
//
System.out.println("第一次比较进行equals比较===》"+ (str1.equals(str2)));

上面的在比较什么为什么==和equals得出来的结果不一样呢?

咱们分开来进行讲述。

1、关系操作符“==”

说的简单一点,==就是比较值是否相同,有一个前提就是进行比较的对象是基本数据类型。说到这里你有没有在自己脑子里面回想那8种基本数据类行呢?🤭 那么接下来我们一起来复习Java最基本的东西,不能好高骛远哦。
在这里插入图片描述

接下来还有三种引用类型。接口,类,数组
示例:

int i=2;
int m=2;
System.out.println(i==m);//基本数据类型比较的是值。
String st=new String("qqq");
String str1=new String("aaa");
String str2=new String("aaa");

System.out.println("第一次比较:"+(str1==str2));

System.out.println("第一次比较进行equals比较===》"+ (str1.equals(str2)));

str1=st;
str2=st;

System.out.println("交换之后进行比较"+(str1==str2));

这样你会发现第三次输出和第四次的输出是一样的。因为str1str2都引用了同一个对象,所以你会发现==如果用于引用类型的话比较的是对象的内存地址。

2、方法equals源码刨析

equals方法原本是Object类中定义的,因为Object是所有类的父类,所以其他的类也可以重写equals方法,(你有没有忘记equals本身是那个类里面的呢?🤭)

equals本身比较的是对象的地址。

Object源码如下:

   public boolean equals(Object obj) {
        return (this == obj);//比较内存地址
    }

因为上面介绍过==比较引用类型比较的是内存地址,所以说equals方法本身比较的是内存地址,但是因为很多类重写了父类的equals方法,把equals方法进行了改造,所以说,有点equals方法比较的是对象的值。

String源码如下:

public boolean equals(Object anObject) {
        if (this == anObject) {
            return true;
        }
        /* 重写父类方法添加判断条件,比较其值 */
        if (anObject instanceof String) {
            String anotherString = (String)anObject;
            int n = value.length;
            if (n == anotherString.value.length) {
                char v1[] = value;
                char v2[] = anotherString.value;
                int i = 0;
                while (n-- != 0) {
                    if (v1[i] != v2[i])
                        return false;
                    i++;
                }
                return true;
            }
        }
        return false;
    }

你可以看到相比较而言比父类中多了一个判断,使用数组判断每个Char是否相等,也就是判断其值是否相等,所以说如果内存地址不一样,但值是一样的,也可以返回true

所以说不要傻乎乎的只回答equals比较的是值。把原理给他说出来会是很棒的点😄。但是有一点equals不能判断基本的数据类型

接下来我们来看一下其他类重写的equals方法源码。

Integer类中equals方法

 public boolean equals(Object obj) {
        if (obj instanceof Integer) {
            return value == ((Integer)obj).intValue();
        }
        return false;
    }

该方法单纯判断其值是否相等。其他基本数据类型的源码类似就不一一展示了。

Date时间类的源码:

public boolean equals(Object obj) {
    return obj instanceof Date && getTime() == ((Date) obj).getTime();
}

...

public long getTime() {
        return getTimeImpl();
    }

private final long getTimeImpl() {
    if (cdate != null && !cdate.isNormalized()) {
        normalize();
    }
    return fastTime;
}

可以看出比较的也是基本数据类型,原理就是他们把一个引用类型变成一个基本数据类型进行比较

3、总结

  • 对于==,如果作用于基本数据类型的变量,则直接比较其存储的 “值”是否相等。
    • 如果作用于引用类型的变量,则比较的是所指向的对象的地址。
  • 对于equals方法,注意:equals方法不能作用于基本数据类型的变量。
    • 如果没有对equals方法进行重写,则比较的是引用类型的变量所指向的对象的地址;
    • 如果重写后根据具体重写的逻辑判断,来判断是比较其值,还是他们的内存地址。(一般重写后都是比较值)

结束: 通过这一篇博客我想我已经牢牢的掌握了他们的区别,那么身为读者的你们呢?如果以上有何问题欢迎提出指正,也可在留言区交流一下你们面试的囧事😊。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值