首页 > 其他分享 >手撕HashMap

手撕HashMap

时间:2023-03-27 21:44:39浏览次数:57  
标签:return HashMap value hashcode 键值 key

HashMap基本了解


1、 jdk1.7之前,HashMap底层只是数组和链表
2、 jdk1.8之后,HashMap底层数据结构当链表长度超过8时,会转为红黑树
3、 HashMap利用空间换时间的思想,将键值对一个个散落在集合中
4、 hashcode值通过调用hashcode()方法得到,所以有可能存在hashcode值相同的情况,即所谓的哈希冲突
5、手撕hashmap的思路:
6、存储put():

  • Map有一个封装的内部接口Entry<K,V>,用来将key和value封装成键值对对象
  • 键值对对象根据计算的hashcode值进行存储
  • hashmap的特点
    • key不能重复
    • 当key重复时,会把原有的键值对替换成新的键值对

7、取值get():

  • 先调用hashcode方法进行计算,判断是否存在
    • 存在则在链表中进行equals方法一一比较他们的key
    • key值不重复

手撕HashMap

1、首先需要一个Map接口,其中定义我们的put()、get()、hashcode()等方法

public interface FakeMap<K,V> {

    /**
     * 将键值对存入我们自己实现的FakeMap中
     * @param key    传入的key
     * @param value  key所对应的值
     */

    void put(K key,V value);

    /**
     * 通过传入key来获取对应的值
     * @param key 传入的key
     * @return    返回key对应的值,没有则返回null
     */
    V get(K key);

    /*
        说明:
        hashmap所使用的hashcode方法应该来自于key本身提供
        1、我们在模拟hashmap时,需要保证hashcode值的范围,不能超过数组的下表
        2、jdk1.8之后接口内可以写静态方法和default方法
        3、如果两个对象的hashcode相同,对象不一定相同;
            如果对象相同,hashcode一定相同
     */

    /**
     * 自己定义的hashcode方法
     * @param key 需要计算hashcode值的key
     * @return  int类型的hashcode值,人为地将值限制在了0-1999
     */
    default int hashcode(K key){
        return key.toString().hashCode()%2000;
    }

}

2、需要一个Entry类,用来用我们传入的键值对生成键值对对象

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

@Data
@NoArgsConstructor
@AllArgsConstructor
public class Entry<K, V> {
    private K key;
    private V value;
}

3、需要一个Map的实现类,用来实现Map中的各个方法

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;

public class FakeHashMap<K, V> implements FakeMap<K, V> {

    /**
     * 定义一个数组,数组的下标和hashcode值对应,用来存放链表的地址
     */
    LinkedList<Entry<K, V>>[] mapArr = new LinkedList[2000];


    /**
     * 不需要遍历数组,大大减少了代码量,直接存入hashcode的值
     * 用来记录被使用的hashcode,方便后续其他方法的操作
     */
    List<Integer> hashcodeList=new ArrayList<>();
    

    /**
     * 将键值对存入我们自己实现的FakeMap中
     * @param key    传入的key
     * @param value  key所对应的值
     */
    @Override
    public void put(K key, V value) {
        //根据key计算出key对应的hashcode值
        int hashcode = hashcode(key);
        //hashcode值对应的是数组的下标,应该先判断下标对应的链表是否存在,不存在就先创建
        if (null == mapArr[hashcode]) {
            //创建一个链表,并且将我们的key和value封装成键值对对象Entry并存入链表
            Entry<K, V> entry = new Entry(key,value);
            //链表的内存地址存入数组对应的下标处
            mapArr[hashcode] = new LinkedList<>();
            mapArr[hashcode].add(entry);
            hashcodeList.add(hashcode);
        } else {
            //链表存在说明之前已经有键值对存入,需要我们进行判断
            //需要遍历这个链表:1、如果找到key相同的,则更新链表替换 2、如果没有找到,直接新建对象存入
            boolean found = false;
            loop:
            for (Entry<K, V> entry : mapArr[hashcode]
            ) {
                if (entry.getKey().equals(key)) {
                    entry.setValue(value);
                    found = true;
                    //若找到则退出循环
                    break loop;
                }
            }
            if (!found) {
                mapArr[hashcode].add(new Entry<K, V>(key, value));
            }
        }
    }


    /**
     * 通过传入key来获取对应的值
     * @param key 传入的key
     * @return    返回key对应的值,没有则返回null
     */
    @Override
    public V get(K key) {
        int hashcode = hashcode(key);
        //如果发现没存过,直接返回空
        if (null == mapArr[hashcode]) {
            return null;
        } else {
            //如果遍历能查找到key,则根据key取出对应的下标的值,返回value
            //如果遍历不能找到,则返回null
            for (Entry<K, V> entry : mapArr[hashcode]
            ) {
                if (entry.getKey().equals(key)) {
                    return entry.getValue();
                }
            }
        }
        return null;
    }


}

4、最后写一个测试类,测试我们自己手搓的hashmap

  • 传入三个键值对,其中前两个的key值相同,看看是否会自己更新value值
public class Test {
    public static void main(String[] args) {
        FakeMap<String, String> fm = new FakeHashMap<>();
        fm.put("ikun","zhangsan");
        fm.put("ikun","lisi");
        fm.put("boy","wangwu");
        System.out.println(fm.get("ikun"));
        System.out.println(fm.get("boy"));
    }
}

5、测试结果:可以发现lisi替换掉了同样是ikun的zhangsan

6、补充:HashMap还有很多其他的方法,我这里没有全部手撕下来,但是可以根据put和get的思路来做

  • 具体实现的话就是在接口中定义新的方法,并且在实现类中实现再去测试就完事了
/**
     * 删除传入的key值所对应的键值对对象
     *
     * @param key 传入的key
     */
    void remove(K key);

    /**
     * 清除 HashMap 中的所有关联或者映射
     */
    void clear();

    /**
     * 判断是否存在key值所对应的映射,返回一个布尔值
     *
     * @param key 传入一个key的值
     * @return 判断是否存在key值所对应的映射,返回一个布尔值
     */
    boolean containsKey(K key);

    /**
     * 获取HashMap的键的集合,以Set<K>保存
     *
     * @return 返回key的集合
     */
    Set<K> keySet();

    /**
     * 得到 HashMap 中各个键值对映射关系的集合
     *
     * @return 返回一个映射关系的集合
     */
    Set<Entry<K, V>> entrySet();

    /**
     * 将指定所有的键值对插入到 HashMap 中
     *
     * @param fakeMap 包含插入到 HashMap 的映射关系
     */
    void putAll(FakeMap<K, V> fakeMap);

    /**
     * 得到 HashMap 键值对的数量
     *
     * @return 一个int型整数
     */
    int size();

    /**
     * 获取HashMap中value的集合
     *
     * @return 返回value集合
     */
    Collection<V> values();

标签:return,HashMap,value,hashcode,键值,key
From: https://www.cnblogs.com/lynnier/p/17263086.html

相关文章

  • HashMap和LinkedHashMap遍历机制
    原文链接:HashMap和LinkedHashMap遍历机制对HashMap和LinkedHashMap遍历的几种方法以HashMap为例,LinkedHashMap方法一样。一共有三种遍历方式Iterator<Map.Entry......
  • Java数据结构 HashMap 哈希表定义使用
    1.HashMapHashMap是一个散列表,它存储的内容是键值(key-value)映射。其中key和value类型可以相同也就而已不同,根据定义。2.HashMap使用1)定义HashMap<Integer,String>hashmap1......
  • 为什么HashMap查找比List快很多?
    做两数之和这道题目时,引发了一个思考:为什么两者运行时间相差如此之大???好残忍,我List比你HashMap到底差在哪****于是我一顿查资料....战犯哈希算法登场哈希算法......
  • Java中Map类型数据使用LinkedHashMap保留数据的插入顺序
    场景Vue中JS遍历后台JAVA返回的Map数据,构造对象数组数据格式:Vue中JS遍历后台JAVA返回的Map数据,构造对象数组数据格式_BADAO_LIUMANG_QIZHI的博客在上面构造以时间为Key,以......
  • HashMap底层源码分析
    HashMap底层源码分析今天先简单看看HashMap的底层源码,之后做详细的分析以及与其他集合的对比。1.看源码之前需要了解的一些内容Node<K,V>[]table哈希表结构中数组......
  • hashmap,hashtabl,hashtree,linkedhashmap区别分析
    java为数据结构中的映射定义了一个接口java.util.Map;它有四个实现类,分别是HashMapHashtableLinkedHashMap和TreeMap.Map主要用于存储健值对,根据......
  • Java HashMap为什么线程不安全
    一、学习目标1、HashMap线程不安全原因:原因:JDK1.7中,由于多线程对HashMap进行扩容,调用了HashMap#transfer(),具体原因:某个线程执行过程中,被挂起,其他线程已经完成数据迁......
  • hashmap get、put时间复杂度
    在JDK8之前用单链表HashMap作为一个桶来储存存在哈希碰撞的元素。无论是get还是put方法,步骤都可以分为第一步找桶(找桶时间都为O(1),可以忽略),第二步在桶内进行操作(查找或......
  • JUC源码学习笔记8——ConcurrentHashMap源码分析1 如何实现低粒度锁的插入,如何实现统
    源码基于jdk1.8这一片主要讲述ConcurrentHashMap如何实现低粒度锁的插入,如何实现统计元素个数,如何实现并发扩容迁移系列文章目录和关于我一丶ConcurrentHashMap概述......
  • ConcurrentHashMap 的实现方式?
    ConcurrentHashMap的实现方式和Hashtable不同,不采用独占锁的形式,更高效,其中在jdk1.7和jdk1.8中实现的方式也略有不同。Jdk1.7中采用分段锁和HashEntry使锁更加......