内存泄漏避雷!你真的了解重写equals()和hashcode()方法的原因吗?

津津乐道
• 阅读 1321

基本概念

  • 要比较两个对象是否相等时需要调用对象的equals() 方法:

    • 判断对象引用所指向的对象地址是否相等
  • 对象地址相等时, 那么对象相关的数据也相等,包括:

    • 对象句柄
    • 对象头
    • 对象实例数据
    • 对象类型数据
  • 可以通过比较对象的地址来判断对象是否相等

    Object源码

  • 对象在不重写的情况下使用的是Object中的equals() 方法和hashCode() 方法

    • equals(): 判断的是两个对象的引用是否指向同一个对象
    • hashCode(): 根据对象地址生成一个整数数值
  • ObjecthashCode() 方法修饰符为native: 表明该方法是由操作系统实现. Java调用操作系统底层代码获取Hash

    public native int hashCode();

    重写equals

  • 重写equals()方法的场景:

    • 假设现在有很多学生对象
    • 默认情况下,要判断多个学生对象是否相等,需要根据地址判断:

      • 若对象地址相等,那么对象实例的数据一定是一样的
    • 判断相等的要求:

      • 当学生的姓名,年龄,性别相等时,认为对象是相等的,
      • 不一定需要对象的地址完全相同
  • 根据需求重写equals()方法:

    public class Student {
      /** 姓名 */
      private String name;
      /** 性别 */
      private String sex;
      /** 年龄 */
      private String age;
      /** 体重 */
      private float weight;
      /** 地址 */
      private String addr;
    
      /*
       * 重写equals()方法
       */
      @Override
      public boolean equals(Object obj) {
          // instanceof已经处理了obj == null的情况
            if (! (Object instanceof Student)) {
                
                return false;
            }
            Student stuObj = (Student) obj;
            // 地址相等
            if (this == stuObj) {
                return true;
            }
            // 如果对象的姓名,年龄,性别相等.则两个对象相等
            if (stuObj.name.equals(this.name) && stuObj.sex.equals(this.sex) && stuObj.age.equals(this.age)) {
                return true;
            } else {
                return false;
            }
       }
    
       public String getName() {
            return name;
       }
       public void setName(String name) {
            this.name = name;
       }
       public String getSex() {
            return sex;
       }
       public void setSex(String sex) {
            this.sex = sex;
       }
        public String getAge() {
            return age;
       }
       public void setAge(String age) {
            this.age = age;
       }
       public String getWeight() {
            return weight;
       }
       public void setName(String weight) {
            this.weight = weight;
       }
       public String getAddr() {
            return addr;
       }
       public void setAddr(String addr) {
            this.addr = addr;
       }
    }
  • 示例:

    public static void main(String[] args) {
      Student s1 = new Student();
      s1.setAddr("earth");
      s1.setAge("20");
      s1.setName("Tom");
      s1.setSex("Male");
      s1.setWeight(60f);
    
      Student s2 = new Student();
      s2.setAddr("Mars");
      s2.setAge("20");
      s2.setName("Tom");
      s2.setSex("Male");
      s2.setWeight(70f);
    
      if (s1.equals(s2)) {
          System.out.println("s1 == s2");
      } else {
          System.out.println("s1 != s2");
      }
    }
  • 重写了equals() 方法后,这里会输出 [s1==s2]
  • 如果没有重写 equals() 方法,那么必定会输出 [s1!=s2]

    重写hashCode

  • 根据重写equals的方法,上述s1和s2认为是相等的
  • Object中的hashCode()方法:

    • equals() 方法没被修改的前提下,多次调用同一个对象的hashCode() 方法返回的值必须是相同的正数
    • 如果两个对象互相equals(), 那么这两个对象的hashcode值必须相等
    • 为不同的对象生成不同的hashcode可以提升Hash表的性能
  • 重写hashCode()方法:

    public class Student {
    /* 姓名 /
    private String name;
    /* 性别 /
    private String sex;
    /* 年龄 /
    private String age;
    /* 体重 /
    private float weight;
    /* 地址 /
    private String addr;

    /*

    • 重写hashCode()方法
      */

    @Override
    public int hashCode() {

       int result = name.hashCode();
       result = 17 * result + sex.hashCode();
       result = 17 * result + age.hashCode();
       return result;

    }

    /*

    * 重写equals()方法
    */

    @Override
    public boolean equals(Object obj) {

        // instanceof已经处理了obj == null的情况
        if (! (Object instanceof Student)) {
            
            return false;
        }
        Student stuObj = (Student) obj;
        // 地址相等
        if (this == stuObj) {
            return true;
        }
        // 如果对象的姓名,年龄,性别相等.则两个对象相等
        if (stuObj.name.equals(this.name) && stuObj.sex.equals(this.sex) && stuObj.age.equals(this.age)) {
            return true;
        } else {
            return false;
        }

    }

    public String getName() {

        return name;

    }
    public void setName(String name) {

        this.name = name;

    }
    public String getSex() {

        return sex;

    }
    public void setSex(String sex) {

        this.sex = sex;

    }
    public String getAge() {

        return age;

    }
    public void setAge(String age) {

        this.age = age;

    }
    public String getWeight() {

        return weight;

    }
    public void setName(String weight) {

        this.weight = weight;

    }
    public String getAddr() {

        return addr;

    }
    public void setAddr(String addr) {

        this.addr = addr;

    }
    }

  • 在两个对象相等的情况下,分别放入Map和Set中:

    public static void main(String[] args) {
      Student s1 = new Student();
      s1.setAddr("earth");
      s1.setAge("20");
      s1.setName("Tom");
      s1.setSex("Male");
      s1.setWeight(60f);
    
      Student s2 = new Student();
      s2.setAddr("Mars");
      s2.setAge("20");
      s2.setName("Tom");
      s2.setSex("Male");
      s2.setWeight(70f);
    
      if (s1.equals(s2)) {
          System.out.println("s1 == s2");
      } else {
          System.out.println("s1 != s2");
      }
      
      Set set = new HashSet();
      set.add(s1);
      set.add(s2);
      System.out.println(Set);
    }
  • 如果没有重写ObjecthashCode() 方法,会出现:

    [com.oxford.Student@7852e922, com.oxford.Student@4e25154f]
  • 这是不符合预期的,因为Set容器有去重的特性.相等的元素不会重复显示.这就涉及到Set的底层实现了
  • HashSet底层实现:

    • HashSet底层是通过HashMap实现的
    • 比较Set容器内元素是否相等是通过比较对象的hashcode来判断是否相等的
  • hashCode()的写法:

    • 首先整理出判断对象相等的属性
    • 然后去一个尽可能小的正整数,防止最终结果超出整型int的取数范围
    • 然后计算[正整数 * 属性的hashCode + 其余某个属性的hashCode]
    • 重复步骤
    /*
     * 重写hashCode()方法
     */
    @Override
    public int hashCode() {
      int result = name.hashCode();
      result = 17 * result + sex.hashCode();
      result = 17 * result + age.hashCode();
      return result;
# 原理分析
- 因为没有重写父类的**Object**的**hashCode()** 方法,所以**Object**的**hashCode()** 方法会根据两个对象的地址生成响应的**hashcode**
- 由于两个对象分别是实体类创建的不同的实例,所以地址肯定是不一样的,那么**hashcode**值也是不一样的
- **Set区别对象是不是唯一的标准:**
  - 两个对象的**hashcode**值是否一样
  - 然后再判定两个对象是否**equals** 
- **Map区别对象是不是唯一的标准:**
  - 先根据**Key**值的**hashcode**分配来获取保存数组下标
  - 然后再根据**eaquals**区分是否是唯一值 
# HashMap
### HashMap组成结构
- **HashMap:** 是由**数组**和**链表**组成的
### HashMap的存储
- **HashMap的存储:**
  - 一个对象存储到**HashMap**中的位置是由**key**的**hashcode**值决定的 
  - **HashMap查找key:**
    - 查找**key**时 **,hashMap**会先根据**key**值的**hashcode**经过取余算法定位所在数组的位置
    - 然后根据**key**的**equals**方法匹配相同的**key**值获取相应的对象 
- **存值规则:**
  - 将**Key**的**hashcode**与**HashMap**的容量,进行**取余**运算得出该**Key**存储在数组所在位置的下标
- **HashMap查找key:**
  - 得到**key**在数组中的位置
  - 匹配得到对应**key**值对象
  - 然后将上述多个对象根据**key.equals()** 来匹配获取对应的key的数据对象
- **HashMap中的hashCode:** 
  - 如果没有**hashcode**就意味着**HashMap**存储的时候是没有规律可循的
点赞
收藏
评论区
推荐文章
blmius blmius
4年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Wesley13 Wesley13
3年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
美凌格栋栋酱 美凌格栋栋酱
7个月前
Oracle 分组与拼接字符串同时使用
SELECTT.,ROWNUMIDFROM(SELECTT.EMPLID,T.NAME,T.BU,T.REALDEPART,T.FORMATDATE,SUM(T.S0)S0,MAX(UPDATETIME)CREATETIME,LISTAGG(TOCHAR(
Wesley13 Wesley13
3年前
java 复制Map对象(深拷贝与浅拷贝)
java复制Map对象(深拷贝与浅拷贝)CreationTime2018年6月4日10点00分Author:Marydon1.深拷贝与浅拷贝  浅拷贝:只复制对象的引用,两个引用仍然指向同一个对象
仔细看看,会有收获。js深浅拷贝
好好理解深浅拷贝和赋值(针对引用类型)赋值:两个对象指向同一内存地址。结果,无论是修改基本类型还是引用类型,两个对象的值都会改变。浅拷贝:两个对象指向不同的内存地址,但是他们中的引用类型数据指向同一内存地址。结果,修改引用类型,两个对象的值都会改变;修改基本类型,互不影响。深拷贝:两个对象指向不同的内存地址,他们中的引用类型也指向不同的内存地址。结果,均互不
Wesley13 Wesley13
3年前
java常用类(2)
三、时间处理相关类Date类:计算机世界把1970年1月1号定为基准时间,每个度量单位是毫秒(1秒的千分之一),用long类型的变量表示时间。Date分配Date对象并初始化对象,以表示自从标准基准时间(称为“历元”(epoch),即1970年1月1日08:00:00GMT)以来的指定毫秒数。示例:packagecn.tanjian
Wesley13 Wesley13
3年前
java中字符串相等判断
字符串的判断有2种:        1、判断地址是否相等 用:        2、判断值是否相等 用:equals方法Object类作为所有类的超类,而Object类的equals方法是直接比较地址的,源码如下:publicbooleanequals(Objectobj){
红烧土豆泥 红烧土豆泥
4年前
补充关于equals的比较方式
补充(equals比较)Object中的equals比较的是地址languagepublicbooleanequals(Objectobj)return(thisobj);java.lang.String类中equals的方法equals判断相等依据策略:如果与目标相等返回0,小于目标返回值小于0,大于目标返回值大于0lan
Stella981 Stella981
3年前
Sonar 规则
bug类型:1、".equals()"shouldnotbeusedtotestthevaluesof"Atomic"classes.bug主要不要使用equals方法对AtomicXXX进行是否相等的判断Atomic变量永远只会和自身相等,Atomic变量没有覆写equals()方法.2、""
Wesley13 Wesley13
3年前
unity将 -u4E00 这种 编码 转汉字 方法
 unity中直接使用 JsonMapper.ToJson(对象),取到的字符串,里面汉字可能是\\u4E00类似这种其实也不用转,服务器会通过类似fastjson发序列化的方式,将json转对象,获取对象的值就是中文但是有时服务器要求将传参中字符串中类似\\u4E00这种转汉字,就需要下面 publ
Stella981 Stella981
3年前
JVM02
文章目录前言对象创建1.类加载检查2.分配内存分配内存的方式内存分配的并发问题3.初始化零值4.设置对象头:5\.执行init方法;对象内存布局对象头实例数据对齐填充对象访问方式使用句柄