手写一个简单的HashMap

HashMap

Hash:Hash散列将一个任意的长度通过某种(hash算法)算法转换成一个固定值。 移位
Map:地图x,y 存储

工作原理:通过hash算法,通过put和get存储和获取对象。

存储对象时,我们将K/V传给put方法时,它调用hashCode计算hash从而得到bucket位置,进一步存储,HashMap会根据当前bucket的占用情况自动调整容量(超过Load Factor则resize为原来的2倍)。如果发生碰撞的时候,Hashmap通过链表将产生碰撞冲突的元素组织起来。如果一个bucket中碰撞冲突的元素超过某个限制(默认是8),则使用红黑树来替换链表,从而提高速度。

获取对象时,我们将K传给get,它调用hashCode计算hash从而得到bucket位置,并进一步调用equals()方法确定键值对。

总结:通过hash出来的值,然后通过值定位到这个map,然后value存储到这个map中hashmap基本原理

手写源码

1.map接口

下面是我手写的hashmap

package map;
public interface Map<K,V> {
public V put(K k,V v);
public V get(K k);
public int size();
public interface Entry<K,V>{
	public  K setKey();
	public V getValue();
}
}

2.实现HashMap

package map;
public class HashMap<K,V> implements Map<K, V> {
private static int defaultLength=16;
private static double defaultLoader=0.75;
private Entry<K,V> [] table=null;
private int size=0;
public HashMap() {
	this(defaultLength,defaultLoader);
}
public HashMap(int length,double loader) {
	defaultLength=length;
	defaultLoader=loader;
	table=new Entry[defaultLength];
}
@Override
public V put(K k, V v) {
	size++;
	int index=hash(k); 
	Entry<K,V> entry=table[index];
	if(entry==null) {//判断是否已经创建
		table[index]=newEntry(k,v,null);
	}else {
		table[index]=newEntry(k,v,entry);
	}
	return table[index].getValue();
}
public Entry<K,V> newEntry(K k, V v, Entry<K,V> next) {
	return new Entry<K,V>(k,v,next);
}
public int hash(K k) {
	int m=defaultLength;
	int i=k.hashCode()%m; //得到哈希值取模
	return i>=0?1:-i;
}
@Override
public V get(K k) {
	int index=hash(k);
	if(table[index]==null) { //判断是否找到
		return null;
	}
	return find(k,table[index]);
	
}
public V find(K k, HashMap<K, V>.Entry<K, V> entry) {
	if(k==entry.getK()||k.equals((entry.getK()))){ //判断key值是否相等
		return entry.getV();
	}else {
		if(entry.next!=null) { //不相等从next里继续往下找
			return find(k,entry.next);
		}
	}
	return null;
}
@Override
public int size() {
	return 0;
}
class Entry<K,V> implements Map.Entry<K, V>{
	K k;
	V v;
	Entry<K,V> next;
	public Entry(K k,V v,Entry<K,V> next) {
		this.k=k;
		this.v=v;
		this.next=next;
	}
	public K getK() {
		return k;
	}
	public void setK(K k) {
		this.k = k;
	}
	public V getV() {
		return v;
	}
	public void setV(V v) {
		this.v = v;
	}
	public Entry<K, V> getNext() {
		return next;
	}
	public void setNext(Entry<K, V> next) {
		this.next = next;
	}
	@Override
	public K setKey() {
		return null;
	}
	@Override
	public V getValue() {
		return null;
	}
}
}

不足之处(伸缩性角度)

Key是否重复相关

1.伸缩性

2.时间复杂度:你的hash算法决定了你的效率

伸缩性角度:每当hashmap扩容的时候需要重新去addentry对象,需要Hash。然后放入我们新的entry table的数组里
如果你们的工作中你知道你们hashmap需要存多少值 几千或几万的时候最好先指定他们的扩容大小 防止在put的时候进行再次扩容很多次

全部评论

相关推荐

从输入URL到页面加载发生了什么:总体来说分为以下几个过程:&nbsp;1.DNS解析&nbsp;2.TCP连接&nbsp;3.发送HTTP请求&nbsp;4.服务器处理请求并返回HTTP报文&nbsp;5.浏览器解析渲染页面&nbsp;6.连接结束。简述了一下各个过程的输入输出作用:以下是对从输入&nbsp;URL&nbsp;到页面加载各过程的输入、输出或作用的一句话描述:DNS&nbsp;解析:&nbsp;输入:用户在浏览器地址栏输入的域名(如&nbsp;www.example.com)。输出:对应的&nbsp;IP&nbsp;地址(如&nbsp;192.168.1.1)。作用:将易于记忆的域名转换为计算机能够识别和用于网络通信的&nbsp;IP&nbsp;地址,以便浏览器与目标服务器建立连接。TCP&nbsp;连接:&nbsp;输入:浏览器获得的服务器...
明天不下雨了:参考一下我的说法: 关键要讲出输入网址后涉及的每一个网络协议的工作原理和作用: 涉及到的网络协议: HTTP/HTTPS协议->DNS协议->TCP协议->IP协议->ARP协议 面试参考回答: 第一次访问(本地没有缓存时): 一般我们在浏览器地址栏输入的是一个域名。 浏览器会先解析 URL、解析出域名、资源路径、端口等信息、然后构造 HTTP 请求报文。浏览器新开一个网络线程发起HTTP请求(应用层) 接着进行域名解析、将域名解析为 IP 地址 浏览器会先检查本地缓存(包括浏览器 DNS 缓存、操作系统缓存等)是否已解析过该域名 如果没有、则向本地 DNS 服务器请求解析; 本地服务器查不到会向更上层的 DNS 服务器(根域名服务器->顶级域名服务器->权威域名服务器询问)递归查询 最终返回该域名对应的 IP 地址。(应用层DNS协议)DNS 协议的作用: 将域名转换为 IP 地址。 由于 HTTP 是基于 TCP 传输的、所以在发送 HTTP 请求前、需要进行三次握手、在客户端发送第一次握手的时候、( 浏览器向服务器发送一个SYN(同步)报文、其中包含客户端的初始序列号。TCP头部设置SYN标志位、并指定客户端端口 同时填上目标端口和源端口的信息。源端口是浏览器随机生成的、目标端口要看是 HTTP 还是 HTTPS、如果是 HTTP 默认目标端口是 80、如果是 HTTPS 默认是 443。(传输层) 然后到网络层:涉及到(IP协议) 会将TCP报文封装成IP数据包、添加IP头部,包含源IP地址(浏览器)和目标IP地址(服务器)。IP 协议的作用: 提供无连接的、不可靠的数据包传输服务。 然后到数据链路层、会通过 ARP 协议、获取目标的路由器的 MAC 地址、然后会加上 MAC 头、填上目标 MAC 地址和源 MAC 地址。 然后到物理层之后、直接把数据包、转发给路由器、路由器再通过下一跳、最终找到目标服务器、然后目标服务器收到客户的 SYN 报文后,会响应第二次握手。 当双方都完成三次握手后、如果是 HTTP 协议、客户端就会将 HTTP 请求就会发送给目标服务器。如果是 HTTPS 协议、客户端还要和服务端进行 TLS 四次握手之后、客户端才会将 HTTP 报文发送给目标服务器。 目标服务器收到 HTTP 请求消息后、就返回 HTTP 响应消息、浏览器会对响应消息进行解析渲染、呈现给用户
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客企业服务