equals方法相关总结

roseneverdie 2018-12-19 原文

equals方法相关总结

先说一下Object类吧:

   这是一个号称为祖宗类的东西,是所有类的父类,也是唯一一个没有父类的类。

   接口不继承object类

   并且Object类存在于java的lang包中,我们都知道存在于lang包中的类我们是可以直接使用的,后面总结的也都是这方面的

   今天说一下Object类中的equals方法吧

   对于这个方法的总结最先有些懵,借鉴了https://www.cnblogs.com/dolphin0520/category/361055.html这个大神写的一些东西,受益很大,在重写了一些equals方法后也算是彻底掌握了

一.其源码

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

 

二.该方法和“==”的区别

这是来自于《java编程思想》的一段原话:

“关系操作符生成的是一个boolean结果,它们计算的是操作数的值之间的关系”    

简单来说“==”是用来判断两个变量之间是否相等。

equals方法是用来比较地址是否相同的(我们根据上面的源码来解释一下吧)

this==obj      obj是我们传过来的对象地址,this存放的是本类对象的地址

public class Main {

    /**
     * @param args
     */
    public static void main(String[] args) {
        // TODO Auto-generated method stub
        
        int n=3;
        int m=3;
        
        System.out.println(n==m);
        
        String str = new String("hello");
        String str1 = new String("hello");
        String str2 = new String("hello");
        
        System.out.println(str1==str2);
        
        str1 = str;
        str2 = str;
        System.out.println(str1==str2);
    }

}
结果:    true
         false
         true

对于第一个结果相信我们都知道是为什么:因为他直接比较的数值的值是否相等3==3所以返回值一定是true。

对于第二个结果就会有些不理解了:上面不是说“==”比较的是数值是否相同吗?   在这里我们应该了解到引用类型的变量存的并不是数值,他们存的是每一个对象的地址,我们每实例化一个对象就会在堆中重新创建一个对象,其地址不一样.   所以这里因为地址不一样所以返回值为false。

那么对于第三个结果就很好解释了:我们使str1,str2地址相等,所以返回值一定是true。

所以从这里可以看出来“==”比较的是具体数值,但就看你变量存的是啥数值了,可能是地址,或者一个值之类的。

 

三.通过一个代码来说明一些问题

public class Main {

    /**
     * @param args
     */
    public static void main(String[] args) {
        // TODO Auto-generated method stub
        
        String str1 = new String("hello");
        String str2 = new String("hello");
        
        System.out.println(str1.equals(str2));
    }
}
结果:true

 

四.对于上面问题的解释

我们上面说了equals方法比较的是地址,实例化之后地址一定不一样了.但为什么上面程序结果是true

我们应该知道Object是所有类的父类,String类也不例外.可是身为子类,总得有点特权吧(要不然这爹白叫了),没错重写方法,因为在String类中重写了equals方法,所以比较的就不再是地址了,而是传过去的参数.

至于他是怎样比较的,我在下面会将我的比较自己写的简陋但易懂的重写方法写出来

现在这里是java中String类中equals方法的重写

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;
    }

 

五.我重写的equals方法

public class NewLife {
     private int age;
     private String name;
     public NewLife(int age,String  name) {
         this.age=age;
         this.name=name;
     }
     public boolean equals(Object obj){
            if(this==obj) {
                return true;
            }
            else if(obj instanceof NewLife)
            {
                //  Object obj=new NewLife();  此处涉及到了多态
                NewLife nl=(NewLife)obj;
                if(this.age==nl.age)
                {
                    return true;
                }
            }
            return false;
     }
}

public class Test {
  public static void main(String[] args) {
      
        NewLife n1=new NewLife(12,"lihaiya");
        NewLife n2=new NewLife(13,"dazhizhang");
        boolean a=n1.equals(n2);
        System.out.println(a);
}
}

我们能够注意到不论是我的重写,还是系统中的那个重写,都存在有强转这一操作原因如下:

我还是通过代码来说明对象等级的变化吧

NewLife n2=new NewLife(13,"dazhizhang");

1.这里的n2先是子类的一个对象

public boolean equals(Object obj)

2.obj就是我们传过去的n2,为什么要升高他的等级,我们倒也不想,但是我们重写的是父类中的方法,我们能改变的只是其内容,但是其所传参数的类型我们是绝对没有资格改的

   所以这里n2从一个小小的子类对象变成了父类的对象           而且这里涉及到了多态下面就是隐藏的代码了

Object obj=new NewLife();

3.下面是强转的操作了

NewLife nl=(NewLife)obj;

为什么要强转,因为在我重写的方法中,我要比较的是子类中age是否相同,但是根据多态的性质,变量的查找是严格可着父类来的,如果子类中有父类中没有的话,系统是会报错的,所以这里我们必须把Object类型的obj转成子类NewLife类型才行。    

另外还有一个判定是否需要强转的技巧,当我们不确定一个对象是否需要强转时,试着用变量名.的方式调用一下父类中没有,子类中存在的变量试试,报错的话,就要考虑一下等级问题了

六.总结

对于这个方法的理解最重要的还是自己能够重写出来一个equals方法。

因为在重写的过程中很多我们没有注意到或者不想去想的问题就会暴露出来。

发表于 2018-12-19 16:38 *三井寿* 阅读() 评论() 编辑 收藏

 

版权声明:本文为roseneverdie原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/roseneverdie/p/10143763.html

equals方法相关总结的更多相关文章

  1. 常用类

    概述: 一、基本数据类型包装类; 二、字符串相关类;         不可变字符序列:String       […]...

  2. Java 常用类

    常用类 字符串相关的类 String 的特性 String类:代表字符串。Java 程序中的所有字符串字面值( […]...

  3. 常用类

    常用类 常用类 包装类 在类中查看属性方法的快捷键:Ctrl+F12 java.lang包会默认导入每个Jav […]...

随机推荐

  1. 【问题总结】万万没想到,竟然栽在了List手里

    说明 昨天同事开发的时候遇到了一个奇怪的问题。 使用Guava做缓存,往里面存一个List,为了方便描述,称它 […]...

  2. usb-host一步一步学(二)安卓在usb-host模式下列出当前连接的usb设备

    之前写了一个简单的例子usb-host一步一步学(一)安卓在usb-host模式下列出当前连接的usb设备,下 […]...

  3. 【读书笔记】《Effective Java》——创建和销毁对象

    Item 1. 考虑用静态工厂方法替代构造器 获得一个类的实例时我们都会采取一个共有的构造器。Foo x = […]...

  4. 利用kubernetes 安装 Kubernetes Dashboard

    最近再次学习下k8s,版本已经升级到1.17了,准备部署一下,以下是官方地址 https://github.c […]...

  5. Modelsim仿真流程

    Modelsim仿真流程 1、 Modelsim简介 略。 2、 modelsim仿真流程:modelsim基 […]...

  6. 富文本及编辑器的跨平台方案

    本文将围绕富文本跨平台和编辑器跨平台两个部分介绍跨平台的价值,以及如何实现跨平台。通过一些方案介绍和踩坑分享, […]...

  7. (在模仿中精进数据可视化07)星球研究所大坝分布可视化

    本文完整代码及数据已上传至我的Github仓库https://github.com/CNFeffery/Fef […]...

  8. 我把代码开源、托管到了GitHub、码云

      前言   学习了那么多知识点,写了那么多代码,一直都没有时间整理,之前都是新学一个知识点就在同一个工程项目 […]...

展开目录

目录导航