热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

JavaHashMap实现原理0——从hashCode,equals说起

Java集合类中常见的hashSet,hashMap,hashTable(现已很少用,几乎都采用hashMap替代)的实现都离不开散列表,而散列表的优势在于O(1)级别的查找,而has

Java集合类中常见的hashSet,hashMap,hashTable(现已很少用,几乎都采用hashMap替代)的实现都离不开散列表,而散列表的优势在于O(1)级别的查找,而hashCode()是其中的关键支撑。

什么是hashCode()?

hashCode() 是获取哈希码函数,返回一个整数,确定对象在哈希表中的索引位置。hashCode() 定义在Object类中,Java中的任何类都包含有hashCode() 函数,Object类的hashCode()方法返回这个对象存储的内存地址的编号。虽然每个Java类都包含hashCode() 函数。但是在散列表相关的类中才有用(确定对象在散列表中的位置),在其它情况下基本没用。什么是散列表就不介绍了,基础的数据结构

什么是equals()?

很多文章会将equals()和hashCode()放在一起解释,两者的相同点是都在Object类中定义,以及两个对象equals,则两个对象hashCode()相同(推荐的设计原则)等。所以重写equals之后也要重写hashCode()。和equals一起考察的还有==,==常用来和equals比较区别:==判断两个对象是不是同一个对象,即它们的内存地址是否一致;equals比较两个对象是否相等,即是否是同一个类型,对象内容是否一致。所以由==可以推出equals。而equals又可以推出hashCode()相等。

hashCode()遇上hashMap

使用hashMap泛型类时,将其实例化的类,要重写类的equals和hashCode()方法,才能达到我们对hashMap定义的效果。为什么这么说呢?得参考下hashMap的部分源代码。

public V put(K key, V value) {
// HashMap允许存放null键和null值。
// 当key为null时,调用putForNullKey方法,将value放置在数组第一个位置。
if (key == null)
return putForNullKey(value);
// 根据key的hashCode重新计算hash值。
int hash = hash(key.hashCode());
// 搜索指定hash值所对应table中的索引。
int i = indexFor(hash, table.length);
// 如果 i 索引处的 Entry 不为 null,通过循环不断遍历 e 元素的下一个元素。
for (Entry e = table[i]; e != null; e = e.next) {
Object k;
if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {
V oldValue = e.value;
e.value = value;
e.recordAccess(this);
return oldValue;
}
}
// 如果i索引处的Entry为null,表明此处还没有Entry。
// modCount记录HashMap中修改结构的次数
modCount++;
// 将key、value添加到i索引处。
addEntry(hash, key, value, i);
return null;
}

注意到,hashMap中存放健值对的位置,是由key的hashCode经过hash()操作后生成的。hash函数如下

static int hash(int h) {
h ^= (h >>> 20) ^ (h >>> 12);
return h ^ (h >>> 7) ^ (h >>> 4);
}

  从put()代码可以看出hashMap的插入流程:先调用hashCode方法得到该元素的hashCode值,hash()重新计算hashCode,然后查看table中是否存在该hashCode值,如果存在则调用equals方法重新确定是否存在该元素,如果存在,则更新value值,否则将新的元素添加到HashMap中。从这里可以看出,hashCode方法的存在是为了减少equals方法的调用次数,从而提高查找效率。
  hash()函数是一个单调函数,返回值和h参数之间是一一映射的关系。所以如果满足equals()的两个对象,hashCode()不同,在hashMap中会被当作两个不同的对象分别插入,语义上就会造成偏差。下面的代码具体反映了偏差

class People{
private String name;
private int age;
public People(String name,int age) {
this.name = name;
this.age = age;
}
public void setAge(int age){
this.age = age;
}
@Override
public boolean equals(Object obj) {
return this.name.equals(((People)obj).name) && this.age== ((People)obj).age;
}
}
public class Main {
public static void main(String[] args) {
HashMap hashMap = new HashMap();
People p1 = new People("Jack", 12);
//System.out.println(p1.hashCode());
hashMap.put(p1, 1);
System.out.println(hashMap.get(new People("Jack", 12)));
}
}

发现,插入了p1,但是在get()和p1 equals的对象时,返回的却为空。因为new People(“Jack”,12)和p1的hashCode没有在People类中重写,返回的是对象默认的内存地址,而两个对象的内存地址不同,所以hashCode不同,查找hashMap时返回为null。

重写hashCode,equals原则

  1. 同一个对象调用多次,hashCode方法必须始终如一地返回同一个整数。
  2. 如果两个对象equals比较是相等的,那么调用两个对象的hashCode方法必须返回相同的整数结果。
  3. 如果两个对象根据equals比较是不等的,hashCode方法不一定得返回不同的整数。

对于第一点,要防止自定义的hashCode依赖对象自身的字段,比如同一个对象,如果只是修改字段的值,hashCode不应该改变。

遵守以上三点原则,在hashMap(包括hashSet,hashTable)中,就不会出现违背hashMap定义的情况

最后

本文参考了浅谈Java中的hashcode方法,深入Java集合学习系列:HashMap的实现原理。
很惭愧,做了一点微小的贡献!


推荐阅读
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • HashTable与ConcurrentHashMap均可实现HashMap的功能,对外提供了键值对存储的数据结构。但是在内部结构及实现上有何区别,性能上的差异到底在哪里又是如何导致的 ... [详细]
  • 缓存这个东西就是为了提高运行速度的,由于缓存是在寸土寸金的内存里面,不是在硬盘里面,所以容量是很有限的。LRU这个算法就是把最近一次使用时间离现在时间最远的数据删除掉。先说说List:每 ... [详细]
  • Java面试 HashMap、HashSet源码解析
    本章所有源代码基于JDK1.8版本HashMap和HashSet是JavaCollectionFramework的两个重要成员,其中HashMap是Map接口的常用实现类,Hash ... [详细]
  • 如何解决《为什么这个hashCode()方法被认为很差?》经验,为你挑选了1个好方法。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • 深入解析Redis内存对象模型
    本文详细介绍了Redis内存对象模型的关键知识点,包括内存统计、内存分配、数据存储细节及优化策略。通过实际案例和专业分析,帮助读者全面理解Redis内存管理机制。 ... [详细]
  • 一、HashSet1.虑重功能特性(HashMap实现)2.put(key)如果重复返回false***Add ... [详细]
  • 图解HashMap
    什么是HashMap,文章内HashMap源码主要来自Android7.0HashMap是开发中常用的一个类,那么他究竟是什么呢?HashMap是一个存储key-value的集合, ... [详细]
  • 本篇文章给大家分享的是有关Java中怎么对HashMap按键值排序,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话 ... [详细]
  • 将学生对象和学生的归属地通过键与值存储到map集合中。importjava.util.HashMap;importjava.util.Iterator;importjava.uti ... [详细]
  • 01Map集合概述A:Map集合概述:我们通过查看Map接口描述,发现Map接口下的集合与Collection接口下的集合,它们存储数据的形式不同a:Collection中的集合 ... [详细]
  • 写这篇文章起源于一道面试题,如何将自定义的类对象作为key存储到HashMap中,即考虑怎么判断key的唯一性。首先,我们看以下HashMap中put(…)方法的源码:public ... [详细]
  • 你知道一个对象的唯一标志不能仅仅通过写一个漂亮的equals来实现太棒了,不过现在你也必须实现hashCode方法。让我们看看为什么和怎么做才是正确的。相等和哈希码相等是从一般的方面 ... [详细]
  • 本文来自:高爽|Coder,原文地址:http:blog.csdn.netghsauarticledetails16843543,转载请注明。HashMap可以 ... [详细]
author-avatar
Resolve
愿你的生活,既有软肋又有盔甲!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有