准备工作 ,实现自己的Map.entry。代码如下 :

import java.util.Map;
public class MapEntry<K,V> implements Map.Entry<K,V> {
private K key;
private V value; public MapEntry(K key, V value) {
this.key = key;
this.value = value;
} @Override
public K getKey() {
return key;
} @Override
public V getValue() {
return value;
} @Override
public V setValue(V v) {
V result = value;
value = v;
return result;
} public String toString() {
return key + " = " + value;
}
}

1.简单的用一对ArrayList来实现。(为什么不用LinkedList?因为无论是put()还是get()都有查询,还是ArrayList快一点)。

import java.util.*;

public class SimpleHashMap<K, V> extends AbstractMap<K, V> {

    private List<K> keys = new ArrayList<>();
private List<V> values = new ArrayList<>(); public V put(K key, V value) {
V oldvalue = get(key);
if (!keys.contains(key)) {
keys.add(key);
values.add(value);
} else {
values.set(keys.indexOf(key), value);
}
return oldvalue;
} public V get(Object key) {
if (!keys.contains(key)) {
return null;
}
return values.get(keys.indexOf(key));
} @Override
public Set<Entry<K, V>> entrySet() {
Set<Map.Entry<K, V>> set = new HashSet<Entry<K, V>>();
Iterator<K> ki = keys.iterator();
Iterator<V> vi = values.iterator();
while (ki.hasNext())
set.add(new MapEntry<K, V>(ki.next(), vi.next()));
return set;
} public static void main(String args[]) {
SimpleHashMap<String, String> map = new SimpleHashMap<String, String>();
map.put("hello1", "world");
map.put("hello2", "world2");
map.put("hello3", "world3");
System.out.println(map);
System.out.println(map.get("hello"));
System.out.println(map.entrySet());
}
}

这种实现能够满足现有测试工作。但是有2个问题需要考虑:

1.在put()和get()的时候都是用list.contains()来判断,如果是按地址查找,速度会不会快更快?

2.所有的put()和get()都是查找整个list,能不能给所有的object按照某一属性分类,这样在查找的时候是不是可以到对应的分类上去查找,这样就不用查整个list,速度会不会再提高一些?

于是就有了第二种实现自己的hashmap的方法,代码如下:

import java.util.*;

public class SecondHashMap<K, V> extends AbstractMap {
static final int SIZE = 997; LinkedList<MapEntry<K, V>>[] buckets = new LinkedList[SIZE]; public V pub(K key, V value) {
V oldvalue = null;
int index = Math.abs(key.hashCode()) % SIZE;
if (buckets[index] == null)
buckets[index] = new LinkedList<MapEntry<K, V>>();
LinkedList<MapEntry<K, V>> bucket = buckets[index];
MapEntry<K, V> pair = new MapEntry<>(key, value);
boolean found = false;
ListIterator<MapEntry<K, V>> it = bucket.listIterator();
while (it.hasNext()) {
MapEntry<K, V> ipair = it.next();
if (ipair.getKey().equals(pair.getKey())) {
oldvalue = ipair.getValue();
it.set(pair);
found = true;
break;
}
}
if (!found)
buckets[index].add(pair);
return oldvalue;
} public V get(Object key){
int index = Math.abs(key.hashCode())%SIZE;
if(buckets[index] == null) return null;
for(MapEntry<K,V> ipair : buckets[index]){
if(ipair.getKey().equals(key))
return ipair.getValue();
}
return null;
} @Override
public Set<MapEntry<K,V>> entrySet() {
Set<MapEntry<K,V>> set = new HashSet<MapEntry<K,V>>();
for(LinkedList<MapEntry<K,V>> bucket : buckets){
if (bucket ==null) continue;
for (MapEntry<K,V> ipair : bucket){
set.add(ipair);
}
}
return set;
} public static void main(String args[]) {
SimpleHashMap<String, String> map = new SimpleHashMap<String, String>();
map.put("hello1", "world");
map.put("hello2", "world2");
map.put("hello3", "world3");
System.out.println(map);
System.out.println(map.get("hello"));
System.out.println(map.entrySet());
}
}

这样一来,是解决了上面的两个问题。但是对于buckets数组下标的生成仅仅是对key的hashcode与容器的长度做了取模运算。这个buckets数组下标最好分布均匀点,如果都集中在在某一块,hashmap在某些区域负载过重,降低整体的性能。在Efffective Java Programming Language Guide这本书中有关于hashcode生成的指导,有兴趣的同学可以自己找找看看。

另外,补充一个hashmap的知识点:负载因子。负载因子代表当前的存储数/容量。空表是0,半满表示0.5,以此类推。当负载情况达到负载因子的水平时,容器就会自动扩容。实现方式是使容量大致加倍,并重新将现有的对象分布到新的表中,也被称为再散列。hashmap的负载因子默认是0.75,这个因子在时间和空间代价之间达到了平衡。更高的负载因子可以降低所需的空间,但是会增加查找代价,反之亦然。如果知道将要在hashmap中存储多少项,创建一个具有恰当大小的初始容量将可以避免自动再散列的开销。

实现自己的HashMap的更多相关文章

  1. HashMap与TreeMap源码分析

    1. 引言     在红黑树--算法导论(15)中学习了红黑树的原理.本来打算自己来试着实现一下,然而在看了JDK(1.8.0)TreeMap的源码后恍然发现原来它就是利用红黑树实现的(很惭愧学了Ja ...

  2. HashMap的工作原理

    HashMap的工作原理   HashMap的工作原理是近年来常见的Java面试题.几乎每个Java程序员都知道HashMap,都知道哪里要用HashMap,知道HashTable和HashMap之间 ...

  3. 计算机程序的思维逻辑 (40) - 剖析HashMap

    前面两节介绍了ArrayList和LinkedList,它们的一个共同特点是,查找元素的效率都比较低,都需要逐个进行比较,本节介绍HashMap,它的查找效率则要高的多,HashMap是什么?怎么用? ...

  4. Java集合专题总结(1):HashMap 和 HashTable 源码学习和面试总结

    2017年的秋招彻底结束了,感觉Java上面的最常见的集合相关的问题就是hash--系列和一些常用并发集合和队列,堆等结合算法一起考察,不完全统计,本人经历:先后百度.唯品会.58同城.新浪微博.趣分 ...

  5. 学习Redis你必须了解的数据结构——HashMap实现

    本文版权归博客园和作者吴双本人共同所有,转载和爬虫请注明原文链接博客园蜗牛 cnblogs.com\tdws . 首先提供一种获取hashCode的方法,是一种比较受欢迎的方式,该方法参照了一位园友的 ...

  6. HashMap与HashTable的区别

    HashMap和HashSet的区别是Java面试中最常被问到的问题.如果没有涉及到Collection框架以及多线程的面试,可以说是不完整.而Collection框架的问题不涉及到HashSet和H ...

  7. JDK1.8 HashMap 源码分析

    一.概述 以键值对的形式存储,是基于Map接口的实现,可以接收null的键值,不保证有序(比如插入顺序),存储着Entry(hash, key, value, next)对象. 二.示例 public ...

  8. HashMap 源码解析

    HashMap简介: HashMap在日常的开发中应用的非常之广泛,它是基于Hash表,实现了Map接口,以键值对(key-value)形式进行数据存储,HashMap在数据结构上使用的是数组+链表. ...

  9. java面试题——HashMap和Hashtable 的区别

    一.HashMap 和Hashtable 的区别 我们先看2个类的定义 public class Hashtable extends Dictionary implements Map, Clonea ...

  10. 再谈HashMap

    HashMap是一个高效通用的数据结构,它在每一个Java程序中都随处可见.先来介绍些基础知识.你可能也知 道,HashMap使用key的hashCode()和equals()方法来将值划分到不同的桶 ...

随机推荐

  1. classic code review

    package dao; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.ResultSe ...

  2. Ubuntu18.04下make menuconfig缺少ncurses库

    kent@hu:~/work/03-kernel/linux-4.15.1$ make menuconfig *** Unable to find the ncurses libraries or t ...

  3. Python 模块调用的变量与路径

    自己编写的python代码经常需要分模块文件以及包,梳理一下调用顺序.执行顺序.工作路径.函数与变量等 工作路径 首先是工作路径,当模块代码放在统一的包内的时候,其路径和外层的包路径不同,当作为主调用 ...

  4. html2canvas将页面内容生成图片

    html2canvas官网链接地址:https://html2canvas.hertzen.com/ <!doctype html> <html lang="en" ...

  5. Android Studio学习之build.gradle文件

    参考书籍:第一行代码 最外层目录下的build.gradle buildscript{repositories{ jcenter() //代码托管仓库 } dependencies{ classpat ...

  6. miniui的联动列表

    function OnCellBeginEdit(e) { var grid = e.sender; var column = e.column; if (!column.parentfield) { ...

  7. rust 如何搜索,如何debug (解决)

    requirement c程序可以用手动查看.h文件获得定义,或者用dumpbin分析lib或者dll获得二进制信息. 但是rust如何得到库的定义呢? rust如何查看函数定义? rust如何deb ...

  8. python selenium 处理时间日期控件(十六)

    测试过程中经常遇到时间控件,需要我们来选择日期,一般处理时间控件通过层级定位来操作或者通过调用js来实现. 1.首先我们看一下如何通过层级定位来操作时间控件. 通过示例图可以看到,日期控件是无法输入日 ...

  9. webRTC中音频相关的netEQ(一):概述

    上篇文章(语音通信中终端上的时延(latency)及减小方法)说从本篇开始会切入webRTC中的netEQ主题,netEQ是webRTC中音频技术方面的两大核心技术之一(另一核心技术是音频的前后处理, ...

  10. Scrapy实战篇(九)之爬取链家网天津租房数据

    以后有可能会在天津租房子,所以想将链家网上面天津的租房数据抓下来,以供分析使用. 思路: 1.以初始链接https://tj.lianjia.com/zufang/rt200600000001/?sh ...