数据结构03-哈希表

作者: 最爱的火 | 来源:发表于2018-05-06 18:36 被阅读25次

数据结构03-哈希表

一、哈希表介绍

1.由来

我们知道,数组查询容易,插入和删除困难;链表查询困难,插入和删除容易。数组和链表的优缺点刚好互补,将他们结合起来,就有一种寻址容易,插入删除也容易的数据结构。哈希表就是这样一种数据结构。

2.基本概念

哈希表(也叫散列表),是根据关键码值(Key)而直接进行访问的数据结构。给定表M,存在函数f(key),对任意给定的关键字值key,代入函数后若能得到包含该关键字的记录在表中的地址,则称表M为哈希表,函数f(key)为哈希函数(也叫散列函数)。

哈希表是一个节点数组,而一个节点就是一个链表。相当于外层为数组,内层为链表。

装填因子:α= 节点数组的元素个数 / 节点数组的长度。一般为0.6~0.9。

哈希冲突:对不同的关键字可能得到同一散列地址,即k≠k2,而f(k1)=f(k2)

均匀散列函数:若对于关键字集合中的任一个关键字,经散列函数映象到地址集合中任何一个地址的概率是相等的,则称此类散列函数为均匀散列函数。

二、哈希表的实现

1.成员属性

这里手写一个哈希表,基本成员属性如下:

// 默认数组大小16,必须为2的幂
static final int DEFAULT_INITIAL_CAPACITY = 1 << 4;
// 默认填充因子
static final float DEFAULT_LOAD_FACTOR = 0.75f;
// 节点数组
Node<K, V>[] table;
// 节点数组的大小
int threshold;
// 插入键值对的个数
int size;

哈希表的扩容很复杂,每次将节点数组增加一倍,为了提高运算效率,使用二进制加法,所以数组大小必须为2的幂,默认数组大小也必须为2的幂。

装填因子越大,越不容易发生哈希冲突,效率越高,内存也越大,所以一般在0.6~0.9。

2.添加

添加的key必须实现hashCode(),否则不能映射地址。

public V put(K k, V v) {
    //初始化成员属性
    initTable();
    //允许key为null
    if (k == null) {
        return putNullKey(v);
    }
    //获取hash值
    int hash = hash(k);
    //映射到数组下标
    int index = getIndex(hash, table.length);
    //键不可以重复
    for (Node<K, V> node = table[index]; node != null; node = node.next) {
        if (hash == node.hash && (node.key == k || node.key.equals(k))) {
            V old = node.value;
            node.value = v;
            return old;
        }
    }
    //添加到哈希表
    addEntry(hash, k, v, index);
    return null;
}

private void addEntry(int hash, K k, V v, int index) {
    //动态扩容
    if (size >= threshold && table[index] != null) {
        checkSize();
        hash = (k == null) ? 0 : hash(k);
        index = getIndex(hash, table.length);
    }
    //获取指定位置的节点
    Node<K, V> e = table[index];
    Node<K, V> newNode = new Node<K, V>(hash, k, v, e);
    table[index] = newNode;
    size++;
}

3.删除

public V remove(Object key) {
    if (size == 0) {
        return null;
    }
    int hash = (key == null) ? 0 : hash(key);
    int index = getIndex(hash, table.length);
    //先找到其头结点
    Node<K, V> pre = table[index];
    Node<K, V> now = pre;
    //根据key循环查找节点,找到后删除
    while (now != null) {
        Node<K, V> next = now.next;
        if (hash == now.hash && (now.key == key || (now.key != null && now.key.equals(key)))) {
            size--;
            if (pre == now) {
                table[index] = next;
            } else {
                pre.next = next;
            }
            return (now == null) ? null : now.value;
        }
        pre = now;
        now = next;
    }

    return (now == null) ? null : now.value;
}

4.查找

private Node<K, V> getEntry(Object key) {
    int hash = (key == null) ? 0 : hash(key);
    int index = getIndex(hash, table.length);
    for (Node<K, V> node = table[index]; node != null; node = node.next) {
        if (hash == node.hash && (node.key == key || node.key.equals(key))) {
            return node;
        }
    }
    return null;
}

三、HashMap与HashTable的区别

HashMap与HashTable都属于哈希表,他们的区别在面试中经常被问到,这里来总结一下:

  • HashMap异步不安全,效率高;HashTable同步安全,效率低。
  • HashMap允许添加空的key,HashTable不允许。
  • HashMap继承自抽象类AbstractMap,而HashTable继承自抽象类Dictionary(已废弃)。
  • HashMap默认的初始化大小为16,之后每次扩充为原来的2倍;HashTable默认的初始大小为11,之后每次扩充为原来的2n+1。

最后

代码地址:https://gitee.com/yanhuo2008/Common/blob/master/Tool/src/main/java/gsw/tool/datastructure/table/HashMap.java

数据结构与算法专题:https://www.jianshu.com/nb/25128590

喜欢请点赞,谢谢!

相关文章

  • 数据结构03-哈希表

    数据结构03-哈希表 一、哈希表介绍 1.由来 我们知道,数组查询容易,插入和删除困难;链表查询困难,插入和删除容...

  • 数据结构_哈希表(Java)

    在讲解HashMap集合之前,我们先说说一个重要的数据结构---哈希表。哈希表是一种非常优秀数据结构,对哈希表进行...

  • 「Redis源码解读」—数据结构(二)哈希表

    Redis的字典使用哈希表作为底层实现 知识点 1.数据结构 哈希节点 哈希表 类型处理函数 2.哈希 哈希算法 ...

  • Tourist with Data Structure Thir

    探索哈希表 概念 哈希集合:哈希集合是集合数据结构的实现之一,用于存储非重复值。哈希映射 :哈希映射是映射数据结构...

  • 使用JavaScript实现哈希表

    关于哈希表的原理详见我的上一篇文章简单易懂数据结构之哈希表 前言 JavaScript中是有哈希类型的数据结构的,...

  • 程序员,你应该知道的数据结构之哈希表

    哈希表简介 哈希表也叫散列表,哈希表是一种数据结构,它提供了快速的插入操作和查找操作,无论哈希表总中有多少条数据,...

  • MySQL索引和锁

    Mysql索引使用的数据结构主要有BTree索引 和 哈希索引 。对于哈希索引来说,底层的数据结构就是哈希表,因此...

  • 哈希表,字典,数组,链表

    1:哈希表 的数据结构,底层实现原理 底层实现:数组 + 链表 哈希表(Hash table,也叫散列表)...

  • 哈希表与树的入门

    哈希表: 特点: 数组(顺序表):寻址容易 链表:插入与删除容易 哈希表:寻址容易,插入删除也容易的数据结构,也就...

  • 2.6 数据结构 --1.5 哈希表

    数据结构子目录https://www.jianshu.com/p/a344fa483655 哈希表 在了解哈希表之...

网友评论

    本文标题:数据结构03-哈希表

    本文链接:https://www.haomeiwen.com/subject/miwurftx.html