equals()与==的区别与实际应用

equals()与 == 在java编程或系统实际可发中都会用到,但是二者又有实际的区别。千万不要理所当然地认为equals()与==的作用就是一样的,下面就一步步来阐述二者的区别。


phone.java
新建phone类,有两个私有的成员变量price与memory,在构造器中对其进行初始化。

package testPackage;

public class phone {

    private float price;//价格
    private int memory;//内存

    public phone(float price, int memory) {
        super();
        this.price = price;
        this.memory = memory;
    }
}

testPhone.java
新建两个手机对象,两个私有成员变量都赋一样值,用equals()进行比较

package testPackage;

public class testPhone {

    public static void main(String[] args){

        phone p1 = new phone(4000f,16);//4000元,16G内存的手机p1
        phone p2 = new phone(4000f,16);//4000元,16G内存的手机p2

        if(p1.equals(p2)){//如果p1等于p2
            System.out.println("p1与p2相同");
        }
        else{//p1不等于p2
            System.out.println("p1与p2不相同");
        }
    }
}

运行结果:
p1与p2不相同

结果分析:
尽管两个手机对象的私有成员变量全部相同,equals()运行结果显示这两个对象是不相同的。这是为什么呢?
还有,phone类中并没有定义equals()方法,这个equals()方法哪里来的?
相信大家都知道java中的所有类都继承了Object.class,因此,所有类都继承了Object类中的equals()方法,要想知道运行结果的本质原因必须查看Object.class的源码。
打开Object.class的源码找到定义equals()方法的位置:

public boolean equals(Object obj) {
    return (this == obj);
}

发现Object.class给出equals()方法的定义,竟然只是简单地通过==去实现判断。而==在java中用于判断存储在栈中的值是否相等。我们的基本数据类型像int的比较都用==,因为基本数据类型就存放在栈中,同样存放在栈中的,还有引用数据类型的引用,而真正的对象本身则存放在堆中。因此,这里给出的equals()方法就只是比较两个对象在内存中的引用(即地址)是否相等。本例中新建了两个对象,在内存中就新开辟了两个内存空间,引用地址当然是不一样的,程序运行结果当然是不相同。
但是我们想通过两个手机的价格与内存相同就判断手机是相同的,确实实际开发中我们都是这样想的,那应该怎么办呢?重写equals()方法。


phone.java中重写equals()方法:

package testPackage;

public class phone {

    private float price;//价格
    private int memory;//内存

    public phone(float price, int memory) {
        super();
        this.price = price;
        this.memory = memory;
    }

    @Override
    public boolean equals(Object obj){
        if(this == obj){//如果引用地址是相同的,证明是同一个对象,那当然相同
            return true;
        }
        if(obj ==null){//如果对象是空的,没有比较的必要,直接返回false
            return false;
        }
        if(!(obj instanceof phone)){//如果obj不是phone的一个对象实例,直接返回false
            return false;
        }
        phone p = (phone)obj;//编译需要,强制类型转换
        if( this.price==p.price && this.memory==p.memory ){
            return true;
        }
        else{
            return false;
        }   
    }
}

重新运行testPhone.java,查看运行结果:

运行结果:
p1与p2相同

结果分析:
通过重写equals()实现了通过私有成员变量判断两个手机对象是否相等,若prize与memory都相等,则两个手机对象就相等。


看到这里,相信大家对equals()与==的区别都有一定理解了,但是显然还不够,关于二者的区别,还有值得了解的地方。


新建testString.java进行字符串测试

package testPackage;

public class testString {

    public static void main(String[] args){

        String s1 = "我是手机1";
        String s2 = new String("我是手机1");

        System.out.println(s1==s2);
        System.out.println(s1.equals(s2));
    }
}

运行结果:
false
true

结果分析:
在java中,字符串是引用数据类型,以实例对象的形式存在。因此,虽然s1与s2看上去定义方式不同,但是它们都新建了对象,本质其实是一样的。
如上所说,==比较的是引用数据类型在栈中的地址是否相同,s1与s2都新建了两个字符串对象,当然开辟了两个不同的内存空间,结果返回false当然是意料之中的。
但是s1.equals(s2)为什么返回true呢?按上面所说Object.class不也是直接通过==进行判断的吗?结果应该是false吧?这时不妨大胆打开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;
    }

发现String.class给出了对Object.class中equals()方法的重写,重写后equals()要判断两个字符串的字符是否全部相等。
这是修改一下s2的字符再重新测试:

public static void main(String[] args){

        String s1 = "我是手机1";
        String s2 = new String("我是手机2");

        System.out.println(s1==s2);
        System.out.println(s1.equals(s2));
    }

运行结果:
false
false

结果分析:
由于s1与s2的字符不完全相同,因此equals()方法返回false


到这里,给出在实际应用中equals()与==的使用技巧:

  1. 若是两个基本数据类型的比较,例如int a = 1,b = 2;比较a是否等于b。使用==
  2. 若要通过比较成员变量判断两个对象是否相等(对象不是字符串),例如自己写的phone对象,要重写从Object类继承来的equals()方法实现。
  3. 若是比较两个字符串,直接用equals()方法。
  • 6
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值