`
yangyangmyself
  • 浏览: 233806 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

哈希表在JAVA中如何实现

    博客分类:
  • Java
阅读更多
一、 复习一下基础知识
1. 截断低位与抹除高位 写道
如果一个数(二进制形式 n 位)对 2k 整除和取模:
● (1)整除是截断低位(k),保留高位(n-k);
● (2)取模运算是抹除最高比特位(要求 k = n-1);
不妨以 10(1010) 和 8(1000) 为例:
● (1)整除:10/8 == 1
● (2)取模:10%8 == 010 == 2
2. 移位运算 写道
在构建哈希表散列值时,常用移位、异或等操作。
1)左移运算符
左移运算符用“<<”表示,是将运算符左边的对象,向左移动运算符右边指定的位数,并且在低位补零。其实,向左移n 位,就 相当于乘上2 的n 次方。
例如:16 << 2 // 结果 64
2)无符号右移运算符
右移运算符用符号“>>>”表示,是将运算符左边的对象向右移动运算符右边指定的位数,并且在高位补0,其实右移n 位,就相当于除上2 的n 次方。
例如:16 >>>2 // 结果 4
3)右移运算符
带符号的右移运算符用符号“>>”表示,是将运算符左边的运算对象,向右移动运算符右边指定的位数。如果是正数,在高位补零,如果是负数,则在高位补1。
例如:16 >> 2 // 结果 4
-16 >> 2 // 结果 -4
  二、 哈希表
1、Hash定义 写道
哈希表通过记录关键字为自变量的函数(称为哈希函数)得到访记录的存储地址查找,所以在哈希表中进行查找操作时,须用同一哈希函数计算得到待查询记录的存储地址,然后到相应的存储单元去获得相关信息再判定查找是否成功。
根据设计哈希函数H(key)和处理冲突的方法,将一组关键字映射到一个有限连限的地址集(区间)上,并以关键字在地址集中的“像”为记录在表中的存储位置,这种表称为哈希表,这一映射过程为哈希造表或散列,所得的存储位置称为哈希地址或散列地址。
冲突定义:对于某个哈希函数H(K)和两个关键字K1和K2,如果K1<>K2,而H(K1)=H(K2),则称为冲突。具有相同哈希函数值的关键字对该哈希函数来说称为同义词。
一般情况,冲突只能尽可能减少而不能完全避免,因为哈希函数是从关键字集合到地址集合映像。通常关键集合比较大,它的元素包含所有可能的关键字,而地址集合的元素仅为哈希表中的地址值。
对于哈希表,主要考虑两个问题:一是如何构造哈希函数,二是如何解决冲突。
2、Hash构造方法 写道
Hash构造即为关键字与哈希地址映射关系建立。
构造哈希函数的原则是:①函数本身便于计算;②计算出来的地址分布均匀,即对任一关键字k,f(k) 对应不同地址的概率相等,目的是尽可能减少冲突。
哈希函数的构造方法很多,常用的有直接定址法、数字分析法、平方取中法、折叠法、除留余数法、随机数法等。
2.1 数字分析法 写道
如果事先知道关键字集合,并且每个关键字的位数比哈希表的地址码位数多时,可以从关键字中选出分布较均匀的若干位,构成哈希地址。例如,有80个记录,关键字为8位十进制整数d1d2d3…d7d8,如哈希表长取100,则哈希表的地址空间为:00~99。假设经过分析,各关键字中 d4和d7的取值分布较均匀,则哈希函数为:h(key)=h(d1d2d3…d7d8)=d4d7。例如,h(81346532)=43,h(81301367)=06。相反,假设经过分析,各关键字中 d1和d8的取值分布极不均匀, d1 都等于5,d8 都等于2,此时,如果哈希函数为:h(key)=h(d1d2d3…d7d8)=d1d8,则所有关键字的地址码都是52,显然不可取。
2.2 平方取中法 写道
当无法确定关键字中哪几位分布较均匀时,可以先求出关键字的平方值,然后按需要取平方值的中间几位作为哈希地址。这是因为:平方后中间几位和关键字中每一位都相关,故不同关键字会以较高的概率产生不同的哈希地址。
例:我们把英文字母在字母表中的位置序号作为该英文字母的内部编码。例如K的内部编码为11,E的内部编码为05,Y的内部编码为25,A的内部编码为01, B的内部编码为02。由此组成关键字“KEYA”的内部代码为11052501,同理我们可以得到关键字“KYAB”、“AKEY”、“BKEY”的内部编码。之后对关键字进行平方运算后,取出第7到第9位作为该关键字哈希地址,如图8.23所示。
关键字 内部编码 内部编码的平方值 H(k)关键字的哈希地址
KEYA 11050201 122157778355001 778
KYAB 11250102 126564795010404 795
AKEY 01110525 001233265775625 265
BKEY 02110525 004454315775625 315
2.3 除留余数法(Jdk HashMap实现) 写道
假设哈希表长为m,p为小于等于m的最大素数,则哈希函数为
h(k)=k % p ,其中%为模p取余运算。
例如,已知待散列元素为(18,75,60,43,54,90,46),表长m=10,p=7,则有
h(18)=18 % 7=4 h(75)=75 % 7=5 h(60)=60 % 7=4
h(43)=43 % 7=1 h(54)=54 % 7=5 h(90)=90 % 7=6
h(46)=46 % 7=4
此时冲突较多。为减少冲突,可取较大的m值和p值,如m=p=13,结果如下:
h(18)=18 % 13=5 h(75)=75 % 13=10 h(60)=60 % 13=8
h(43)=43 % 13=4 h(54)=54 % 13=2 h(90)=90 % 13=12
h(46)=46 % 13=7
此时没有冲突,如图8.25所示。
0 1 2 3 4 5 6 7 8 9 10 11 12
    54   43 18   46 60   75   90
    /**
     * Retrieve object hash code and applies a supplemental hash function to the
     * result hash, which defends against poor quality hash functions.  This is
     * critical because HashMap uses power-of-two length hash tables, that
     * otherwise encounter collisions for hashCodes that do not differ
     * in lower bits. Note: Null keys always map to hash 0, thus index 0.
     * 获取散列值
     */
    final int hash(Object k) {
        int h = hashSeed;
        if (0 != h && k instanceof String) {
            return sun.misc.Hashing.stringHash32((String) k);
        }
        h ^= k.hashCode();
        // This function ensures that hashCodes that differ only by
        // constant multiples at each bit position have a bounded
        // number of collisions (approximately 8 at default load factor).
        h ^= (h >>> 20) ^ (h >>> 12);
        return h ^ (h >>> 7) ^ (h >>> 4);
    }

    /**
     * Returns index for hash code h.
     * 采用除留余数法,构建哈希表
     */
    static int indexFor(int h, int length) {
        // assert Integer.bitCount(length) == 1 : "length must be a non-zero power of 2";
        return h & (length-1);
    }
2.4 与运算法(Jdk ThreadLocalMap实现) 写道
首先定义2的整次幂哈希表长度M,取出小于等于M的最大素数N(一般为N=M-1)
其次通过算法构建一个散列值V(Jdk定义为:0x61c88647)
最后执行与操作:
index = V & N 或
index = V & (M-1)
为什么使用“与”操作,目的是为了将散列值高位全部归零,只保留低位,用于做哈希表地址(即数组下标)。如以哈希表长度16为例,16-1=15,二进制为表示为:00000000 00000000 00001111,与某散列值做“与”操作如下,结果就是截取了最低的四位值。
00000000 00000000 00001111
& 10011100 00001101 00100101 (散列值)

00000000 00000000 00000101 //高全部归零,只留最低4位
/**
* Construct a new map initially containing (firstKey, firstValue).
* ThreadLocalMaps are constructed lazily, so we only create
* one when we have at least one entry to put in it.
*  采用“与”运算构建哈希表
*/
ThreadLocalMap(ThreadLocal firstKey, Object firstValue) {
    table = new Entry[INITIAL_CAPACITY];
    // firstKey.threadLocalHashCode哈希散列值
    // INITIAL_CAPACITY哈希表长度
    int i = firstKey.threadLocalHashCode & (INITIAL_CAPACITY - 1);
    table[i] = new Entry(firstKey, firstValue);
    size = 1;
    setThreshold(INITIAL_CAPACITY);
}
2.5 分段叠加法 写道
这种方法是按哈希表地址位数将关键字分成位数相等的几部分(最后一部分可以较短),然后将这几部分相加,舍弃最高进位后的结果就是该关键字的哈希地址。具体方法有折叠法与移位法。移位法是将分割后的每部分低位对齐相加,折叠法是从一端向另一端沿分割界来回折叠(奇数段为正序,偶数段为倒序),然后将各段相加。例如:key=12360324711202065,哈希表长度为1000,则应把关键字分成3位一段,在此舍去最低的两位65,分别进行移位叠加和折叠叠加,求得哈希地址为105和907。
2.6 伪随机数法 写道
采用一个伪随机函数做哈希函数,即h(key)=random(key)。
在实际应用中,应根据具体情况,灵活采用不同的方法,并用实际数据测试它的性能,以便做出正确判定。通常应考虑以下五个因素 :
1) 计算哈希函数所需时间 (简单)。
2) 关键字的长度。
3) 哈希表大小。
4) 关键字分布情况。
5) 记录查找频率
三、 哈希表冲突处理方法
3.1 开放定址法 写道
这种方法也称再散列法,其基本思想是:当关键字key的哈希地址p=H(key)出现冲突时,以p为基础,产生另一个哈希地址p1,如果p1仍然冲突,再以p为基础,产生另一个哈希地址p2,…,直到找出一个不冲突的哈希地址pi ,将相应元素存入其中。这种方法有一个通用的再散列函数形式:
Hi=(H(key)+di)% m i=1,2,…,n
其中H(key)为哈希函数,m 为表长,di称为增量序列。增量序列的取值方式不同,相应的再散列方式也不同。主要有以下三种:
线性探测再散列
dii=1,2,3,…,m-1
这种方法的特点是:冲突发生时,顺序查看表中下一单元,直到找出一个空单元或查遍全表。
二次探测再散列
di=12,-12,22,-22,…,k2,-k2 ( k<=m/2 )
这种方法的特点是:冲突发生时,在表的左右进行跳跃式探测,比较灵活。
伪随机探测再散列
di=伪随机数序列。
具体实现时,应建立一个伪随机数发生器,(如i=(i+p) % m),并给定一个随机数做起点。
例如,已知哈希表长度m=11,哈希函数为:H(key)= key % 11,则H(47)=3,H(26)=4,H(60)=5,假设下一个关键字为69,则H(69)=3,与47冲突。如果用线性探测再散列处理冲突,下一个哈希地址为H1=(3 + 1)% 11 = 4,仍然冲突,再找下一个哈希地址为H2=(3 + 2)% 11 = 5,还是冲突,继续找下一个哈希地址为H3=(3 + 3)% 11 = 6,此时不再冲突,将69填入5号单元,参图8.26 (a)。如果用二次探测再散列处理冲突,下一个哈希地址为H1=(3 + 12)% 11 = 4,仍然冲突,再找下一个哈希地址为H2=(3 - 12)% 11 = 2,此时不再冲突,将69填入2号单元,参图8.26 (b)。如果用伪随机探测再散列处理冲突,且伪随机数序列为:2,5,9,……..,则下一个哈希地址为H1=(3 + 2)% 11 = 5,仍然冲突,再找下一个哈希地址为H2=(3 + 5)% 11 = 8,此时不再冲突,将69填入8号单元
/**
         * Set the value associated with key.
         *
         * @param key the thread local object
         * @param value the value to be set
         */
        private void set(ThreadLocal key, Object value) {

            // We don't use a fast path as with get() because it is at
            // least as common to use set() to create new entries as
            // it is to replace existing ones, in which case, a fast
            // path would fail more often than not.

            Entry[] tab = table;
            int len = tab.length;
            int i = key.threadLocalHashCode & (len-1);

            for (Entry e = tab[i];
		 e != null;
		 e = tab[i = nextIndex(i, len)]) {
                ThreadLocal k = e.get();

                if (k == key) {
                    e.value = value;
                    return;
                }

                if (k == null) {
                    replaceStaleEntry(key, value, i);
                    return;
                }
            }

            tab[i] = new Entry(key, value);
            int sz = ++size;
            if (!cleanSomeSlots(i, sz) && sz >= threshold)
                rehash();
        }
3.2 链地址法(Jdk HashMap实现) 写道
这种方法的基本思想是将所有哈希地址为i的元素构成一个称为同义词链的单链表,并将单链表的头指针存在哈希表的第i个单元中,因而查找、插入和删除主要在同义词链中进行。链地址法适用于经常进行插入和删除的情况。
例如,已知一组关键字(32,40,36,53,16,46,71,27,42,24,49,64),哈希表长度为13,哈希函数为:H(key)= key % 13,则用链地址法处理冲突的结果如图8.27所示:


 
static class Entry<K,V> implements Map.Entry<K,V> {
        final K key;
        V value;
        Entry<K,V> next;
        int hash;
        /**
         * Creates new entry.
         */
        Entry(int h, K k, V v, Entry<K,V> n) {
            value = v;
            next = n;
            key = k;
            hash = h;
        }
 /**
     * Adds a new entry with the specified key, value and hash code to
     * the specified bucket.  It is the responsibility of this
     * method to resize the table if appropriate.
     *
     * Subclass overrides this to alter the behavior of put method.
     */
    void addEntry(int hash, K key, V value, int bucketIndex) {
        if ((size >= threshold) && (null != table[bucketIndex])) {
            resize(2 * table.length);
            hash = (null != key) ? hash(key) : 0;
            bucketIndex = indexFor(hash, table.length);
        }

        createEntry(hash, key, value, bucketIndex);
    }
}
3.3 再哈希法 写道
这种方法是同时构造多个不同的哈希函数:
Hi=RH1(key) i=1,2,…,k
当哈希地址Hi=RH1(key)发生冲突时,再计算Hi=RH2(key)……,直到冲突不再产生。这种方法不易产生聚集,但增加了计算时间
3.4 公共溢出区 写道
这种方法的基本思想是:将哈希表分为基本表和溢出表两部分,凡是和基本表发生冲突的元素,一律填入溢出表
  • 大小: 4.8 KB
分享到:
评论
1 楼 woodding2008 2016-11-23  
拜读,收藏

相关推荐

    哈希表java代码

    在Java中,我们通常使用`HashMap`类来实现哈希表,但这里提到的是自定义实现哈希表的Java代码。这个压缩包包含三个文件:`HashTable.java`、`Info.java`和`TestHashTable.java`,分别代表哈希表的实现、存储的数据...

    哈希表在java中如何实现1

    本文将深入探讨哈希表的实现原理,以及在Java中如何构建哈希函数。 首先,我们需要理解哈希函数的基本概念。哈希函数是一个将任意大小的输入(通常是字符串或数字)转化为固定大小输出的函数。在哈希表中,这个输出...

    哈希表相关操作实现

    哈希表,也被称为散列表,是计算机科学中一种非常重要的数据结构,它提供了...无论是编程语言的内置数据结构,如Python的dict或Java的HashMap,还是在数据库系统、缓存机制等应用场景中,哈希表都是不可或缺的一部分。

    拉链法哈希表的设计与实现

    在这个场景中,我们看到一个基于Java的课程设计项目,它利用拉链法实现了一个哈希表,用于电话查询系统的功能。让我们深入了解一下这个主题。 哈希表是一种高效的数据结构,通过将关键字(key)映射到数组的索引...

    Java数据结构 线性表,链表,哈希表是常用的数据结构

    Java数据结构 线性表,链表,哈希表是常用的数据结构,在进行Java开发时,JDK已经为我们提供了一系列相应的类来实现基本的数据结构

    哈希表的设计与实现.rar

    5. **源代码实现**:在实际编程中,哈希表的实现通常涉及C++、Java或Python等语言。在实现时,需要考虑内存管理、线程安全、性能优化等问题。例如,Java中的`HashMap`和C++中的`unordered_map`都是内置的哈希表实现...

    哈希表java实现及简单演示

    在Java中,哈希表的实现主要依赖于`java.util.HashMap`类,它是基于哈希表的Map接口实现。在这个Java版的哈希表演示程序中,我们可能会看到如何使用HashMap类来存储和检索数据,以及如何处理可能出现的哈希冲突。 ...

    哈希表基本原理与Java实现

    内容概要:本文详细介绍了哈希表(Hash Table)这一高效的数据...其他说明:虽然本文提供了一种基于链地址法的简单哈希表实现,但在生产环境中建议优先考虑使用更为成熟可靠的库或内置支持(如Java标准库中的HashMap).

    Java哈希表性能优化

    本文将深入探讨Java哈希表的内部机制,并介绍一种新的哈希表实现策略,该策略能够显著提高哈希表在面对特定数据序列时的性能。 #### 二、Java哈希表特点分析 ##### 2.1 装载因子恒定 Java标准库中的哈希表实现...

    Java_Datastructure.rar_java 哈希_java 哈希表

    在IT领域,尤其是编程中,数据结构是至关重要的概念,它们是存储和组织数据...这个压缩包中的"JAVA 重写数据结构"文件可能包含对这些概念的实例化和实现,通过阅读和实践这些代码,可以深入理解哈希表在Java中的应用。

    哈希表实现简单说明-附代码

    哈希表是一种高效的数据结构,它通过特定的哈希函数将键...例如,Python中的`dict`类型、Java的`HashMap`和C++的`std::unordered_map`都是哈希表的典型实现。理解哈希表的工作原理和优化策略对于提升程序性能至关重要。

    数据结构中哈希表的实现代码

    在实际编程中,常见的哈希表实现如Python的内置`dict`类型、Java的`HashMap`以及C++的`std::unordered_map`,它们都提供了高效的键值对操作。这些库通常已经优化了哈希函数和冲突解决策略,使用者无需关心底层细节。...

    哈希表的原理 数据结构

    在 Java 中,哈希表可以用来实现各种数据结构,如 HashSet、HashMap 等。这些数据结构都使用哈希表来存储和查询数据,从而提高了系统的性能。 哈希表是一种非常有用的数据结构,它的优点是查找速度快、时间算法...

    B+树,哈希表等JAVA版本的JAR包

    jdbm-1.0这个库也可能包含了对哈希表的实现,使得在Java中可以高效地处理数据存储和检索。 在实际应用中,B+树和哈希表各有优势:哈希表适合于快速的随机访问,而B+树则擅长范围查询和大数据量的排序。开发者可以...

    哈希表的实现

    6. **应用实例**:哈希表在实际编程中有广泛应用,如Python的字典、Java的HashMap等都是基于哈希表实现的。它们提供了快速的键值对存取,极大地提高了代码执行效率。 7. **源码分析**:了解哈希表的源码可以帮助...

    哈希表-使用Java开发的哈希表-HashTable.zip

    在Java中,`HashTable`是早期版本(Java 1.0)提供的一种线程安全的哈希表实现。尽管现在已经被`HashMap`所替代,但理解`HashTable`的工作原理和特性仍然对深入理解Java集合框架以及数据结构有重要意义。 哈希表的...

    哈希表搜索算法介绍和java代码实现

    下面是一个简单的Java实现哈希表搜索的示例: ```java import java.util.HashMap; public class HashTableSearch { public static void main(String[] args) { // 创建哈希表 HashMap, String&gt; hashtable = new...

    数据结构 哈希表 哈希算法

    在哈希表的实现中,通常还需要考虑以下因素: 1. **装载因子**:哈希表中已存储元素数量与总容量的比例,它直接影响了冲突的概率和查询效率。装载因子过高会导致更多的冲突,因此通常会在装载因子接近一定阈值时进行...

    哈希表,开放地址法之再哈希代码(JAVA)

    在Java中实现开放地址法中的双哈希代码,我们可以创建一个哈希表类,包含以下核心方法: 1. `put(key, value)`: 插入键值对。首先计算初始位置,如果该位置已存在元素,使用双哈希策略找到下一个空位置。 2. `get...

    哈希表-浙大数据结构课件

    4. 常见的哈希表实现:在实际编程中,C++中的`std::unordered_map`、Java中的`HashMap`以及Python的`dict`都是基于哈希表实现的高效容器。这些数据结构提供了高效的插入、删除和查找操作,广泛应用于软件开发的各个...

Global site tag (gtag.js) - Google Analytics