mysql 生成hashmap_hashmap简单实现

import java.util.AbstractMap;

import java.util.HashSet;

import java.util.LinkedList;

import java.util.ListIterator;

import java.util.Map;

import java.util.Set;

/**

记得很早之前看到过一篇帖子 说一个面试者去面试 某大厂要他当场写一个hashmap,其实今天讲的这个例子,足以应付那场面试,核心思想一应俱全,有些地方还欠完善。。。

这个simpleHashmap原理是如下实现

put:

首先定义一个基于链表的Entry hash桶数组

通过key的hashcode取模数组大小得到对应的hash桶下标

如果链表中没有找到对应的key 则加入当前hash桶中

否则覆盖Entry对象

get:

1.通过key的hashcode取模数组大小得到对应的hash桶

判断hash桶中是否存在Entry对象 有取出;没有返回null.

以下是实际hashmap实现思路:

以Entry[]数组实现的哈希桶数组,用Key的哈希值取模桶数组的大小可得到数组下标。

插入元素时,如果两条Key落在同一个桶(比如哈希值1和17取模16后都属于第一个哈希桶),我们称之为哈希冲突。

JDK的做法是链表法,Entry用一个next属性实现多个Entry以单向链表存放。查找哈希值为17的key时,先定位到哈希桶,然后链表遍历桶里所有元素,逐个比较其Hash值然后key值。

在JDK8里,新增默认为8的阈值,当一个桶里的Entry超过閥值,就不以单向链表而以红黑树来存放以加快Key的查找速度。

当然,最好还是桶里只有一个元素,不用去比较。所以默认当Entry数量达到桶数量的75%时,哈希冲突已比较严重,就会成倍扩容桶数组,并重新分配所有原来的Entry。扩容成本不低,所以也最好有个预估值。

取模用与操作(hash & (arrayLength-1))会比较快,所以数组的大小永远是2的N次方, 你随便给一个初始值比如17会转为32。默认第一次放入元素时的初始值是16。

iterator()时顺着哈希桶数组来遍历,看起来是个乱序

*

*

*/

public class SimpleHashMap extends AbstractMap {

staticfinalintSIZE= 997;

LinkedList>[] buckets = new LinkedList[SIZE];

@Override

public V put(K key, V value) {

V oldValue = null;

int index = Math.abs(key.hashCode()) % SIZE;// 取模

if (buckets[index] == null) {

buckets[index] = new LinkedList>();

}

LinkedList> bucket = buckets[index];// 桶位【槽位】

MapEntry pair = new MapEntry(key, value);

booleanfound= false;

ListIterator> it = bucket.listIterator();

while (it.hasNext()) {

MapEntry iPair = it.next();

if (iPair.getKey().equals(key)) {

oldValue = iPair.getValue();

it.set(pair);

found = true;

break;

}

}

if (!found) {

bucket.add(pair);

}

returnoldValue;

}

@Override

public V get(Object key) {

int index = Math.abs(key.hashCode()) % SIZE;

if (buckets[index] == null) {

returnnull;

}

for (MapEntry iPair : buckets[index]) {

if (iPair.getKey().equals(key)) {

return iPair.getValue();

}

}

returnnull;

}

@Override

public Set> entrySet() {

Set> set = new HashSet>();

for (LinkedList> bucket : buckets) {

if (bucket == null) {

continue;

}

for (MapEntry mpair : bucket) {

set.add(mpair);

}

}

returnset;

}

public static void main(String[] args) {

// SimpleHashMap simpleHashMap=new SimpleHashMap();

// simpleHashMap.putAll(Countries.capitals(25));

// System.out.println(simpleHashMap);

// System.out.println(simpleHashMap.get("ERITREA"));

// System.out.println(simpleHashMap.entrySet());

System.out.println(1 ^ 1); // 两个二进制比较 相同取0 不同为一

}

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值