【java基础 15】java代码中“==”和equals的区别

来源:互联网 发布:文件档案管理系统软件 编辑:程序博客网 时间:2024/06/01 23:27

导读:昨夜闲来无事,和贾姑娘聊了聊java基础,然后就说到了这个“==”和equals的问题,我俩都是以前了解过,也常用这个,但是,昨天说到的时候,又乱了,什么比较地址值,什么判断是否同一个对象,基本数据类型啥的,谁都没个准儿,后来写了点代码验证了一番,今儿个写此博客,纪念和好朋友一起探讨学习的经历!PS:我一直知道我这一路走来,受好朋友的恩惠太多了!

一、三组示例代码

1.1,String类(引用类型)

String str1="test";String str2="test";//true trueSystem.out.println(str1==str2);System.out.println(str1.equals(str2));String str3=new String("test");String str4=new String("test");//false trueSystem.out.println(str3==str4);System.out.println(str3.equals(str4));String str5=str1;//true trueSystem.out.println(str5==str1);System.out.println(str5.equals(str1));String str6=str3;//false trueSystem.out.println(str6==str1);System.out.print(str6.equals(str1));

刚开始,统一的意见是“==”是比较值,equals是比较引用对象,请看下面的经典论断(主要是针对str3和str4):

A:==比值,这两个值都是test,应该是true,可结果是false,后面那个equals比较对象,两个都是new的,应该不一样,可结果是true

B:==比值,因为这两个不是简单类型,无法比较,所以返回false;equals 比较对象,此时str3 和str4新创建了两个字符串对象,这两个对象是一样的,返回true

A:如果是byte、int、boolean、long,float,这些的话,==就是比较值,equals比对象,是吧

B:对

1.2,Integer类(int的包装类,非8种基本类型)

Integer int1=1;Integer int2=1;//true trueSystem.out.println(int1==int2);System.out.println(int1.equals(int2));Integer int3=new Integer(1);Integer int4=new Integer(1);//false trueSystem.out.println(int3==int4);System.out.println(int3.equals(int4));Integer int5=int1;//true trueSystem.out.println(int5==int1);System.out.println(int5.equals(int1));Integer int6=int3;//true trueSystem.out.println(int6==int3);System.out.print(int6.equals(int3));

结果和String类一致!

1.3,int(基本类型)



备注:用简单基本类型,根本无法使用equals方法,只能用==,对于String类或者简单类型的包装类(Integer是int的包装类)==比较的是否是同一个地址(对象),equals比较的是地址值


二、分析equals方法

所有的对象都是继承于Object类,那么首先看Object里面,关于equals的定义:

   * @param   obj   the reference object with which to compare.     * @return  {@code true} if this object is the same as the obj     *          argument; {@code false} otherwise.     * @see     #hashCode()     * @see     java.util.HashMap     */    public boolean equals(Object obj) {        return (this == obj);    }
从这里可以看出,equals事实上比较的是否是同一个对象目标,也就是是否是同一个内存地址。但从这个角度来说,那么就无法解释为什么在代码段1中str3.equals(str4)的结果为true了。因为这两个对象都采用了new 关键字,在堆中,必定会存在两个空间地址。为了解决这个问题,势必要去看看String类对于equals方法的重写:

/**     * Compares this string to the specified object.  The result is {@code     * true} if and only if the argument is not {@code null} and is a {@code     * String} object that represents the same sequence of characters as this     * object.     *     * @param  anObject     *         The object to compare this {@code String} against     *     * @return  {@code true} if the given object represents a {@code String}     *          equivalent to this string, {@code false} otherwise     *     * @see  #compareTo(String)     * @see  #equalsIgnoreCase(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类对于equals方法进行了改写,当String类使用equals的时候,if and only if the argument is not null and isa object that represents the same sequence of characters as this object. 对比Object基类返回true的条件:if this object is the same as the obj argument。 现在再去看代码段1中的str3.equals(str4)的结果为true,就能理解了!

接下来,再看基本类型的包装类中对于equals方法的改写:

   /**     * Compares this object to the specified object.  The result is     * {@code true} if and only if the argument is not     * {@code null} and is an {@code Integer} object that     * contains the same {@code int} value as this object.     *     * @param   obj   the object to compare with.     * @return  {@code true} if the objects are the same;     *          {@code false} otherwise.     */    public boolean equals(Object obj) {        if (obj instanceof Integer) {            return value == ((Integer)obj).intValue();        }        return false;    }
if and only if the argument is not null and is an object that contains the same value as this object. 从这里可以看出,它的改写,并未要求是同一对象,而要求是同一值!

附:更为明显的代码片段

Integer intA=new Integer(1);Integer intB=new Integer(1);Integer intC=1;//false false true trueSystem.out.println(intA==intB);System.out.println(intA==intC);System.out.println(intA.equals(intB));System.out.println(intA.equals(intC));

三、总结

写这篇博客,主要是想纪念一下和好朋友之间的探讨交流,感觉和大家一起交流一些简单又有意思的事儿,真心很好玩。而且感觉我现在还比较喜欢看源码,也就来源于对这些小东西的兴趣和好奇。当然目前的学习程度是远远不够的,正在努力中。对了,以上关于equals的各类源码,属于java 8.

多看代码,多交流,多总结,希望自己保持下去,然后和大家一起成长!


0 0
原创粉丝点击