LinkedHashMap

关于HashMap的数据结构:采用散列表的数据结构,链表+数组,数组Array,所存储的元素并非基本数据类型,而是Entry(内含键值对),包括的数据域(键,值,next),

其计算方法:

key.hashcode()%Array[].length


HashMap虽然是一个很好用的集合,但是而有一个问题,那就是在对HashMap进行迭代访问的过程中,添加元素的顺序可能会和访问的数据不一样,这个时候我们可以选择LinkedHashMap


LinkedHashMap继承了HashMap,是其子类,但与之不同的是,增加了一个双向链表来实现元素迭代的顺序,但是肯定会增加时间和空间的消耗

既然继承了HashMap,那么也是非线程安全的

LinkedHashMap的特点:

1.LinkedHashMap的输入和输出顺序一致

2.key和value可以同时为空

3.key可以重复,但是新的key对应的value值会覆盖掉旧value值(这点和HashMap的哈希桶下元素存放规则相似)

 

import java.util.LinkedHashMap;
import java.util.Map;

class Main {
    public static void main(String[] args) {
       LinkedHashMap<String,Object> map = new LinkedHashMap<>();
       map.put("西安",9000);
       map.put("380000","腾讯");
       map.put("何浩源",21);
       map.put("HeHe",'!');
       for (Map.Entry entry : map.entrySet()){
           System.out.println("key: "+entry.getKey()+"   val:"+ entry.getValue());
       }
    }
}

 


 

LinkedHashMap内部的数据结构也是Entry,它是LinkedHashMap的内部类,并在此基础之上添加了两个别的元素,before 和 after,用于维护链表中Entry的前一个元素和后一个元素

LinkedHashMap的实现就是 HashMap+LinkedList 的实现方式,用HashMap维护数据结构,用LinkedList的方式维护数据插入顺序

反向迭代:

import java.util.*;

 class Test8 {

    public static void main(String[] args) {

        Map<String,Object> linkedHashMap = new LinkedHashMap<>(16);
        for (int i = 0; i < 5; i++) {
            linkedHashMap.put(i + "", i);
        }
        for (Map.Entry entry : linkedHashMap.entrySet()) {
            System.out.println("key:" + entry.getKey() + ",value:" + entry.getValue());
        }

        System.out.println("============================");

        // 使用的是迭代器 ListIterator
        ListIterator<Map.Entry> i=
                new ArrayList<Map.Entry>(
                        linkedHashMap.entrySet()).listIterator(linkedHashMap.size());
        while(i.hasPrevious()) {
            Map.Entry entry=i.previous();
            System.out.println("key:" + entry.getKey() + ",value:" + entry.getValue());
        }



    }
}

利用LinkedHashMap实现LRU算法缓存:

http://www.cnblogs.com/xrq730/p/5030920.html

解释一下LRU:LRU即Least Recently Used,最近最少使用,也就是说,当缓存满了,会优先淘汰那些最近最不常访问的数据。
要实现这个功能,就是要记录那些元素被访问了,访问的元素应该向前排列,当然这个功能LinkedHashMap已经帮我们实现了,

 

注意红框的部分, recordAccess,就是将当前entry移动到链表的最前面。

 

 

 

前面说了,LinkedHashMap添加新的元素时会执行删除eldest entry,虽然LinkedHashMap没有实现,但是我们可以重写removeEldestEntry的方法来实现这个功能。

 

public class LRUCache extends LinkedHashMap
{
    public LRUCache(int maxSize)
    {
        super(maxSize, 0.75F, true);
        maxElements = maxSize;
    }

    protected boolean removeEldestEntry(java.util.Map.Entry eldest)
    {
        return size() > maxElements;
    }

    private static final long serialVersionUID = 1L;
    protected int maxElements;
}

当元素的个数大于指定的值时,就会有原始数据被删除掉,也就是链表最后的元素。 

 

posted @ 2019-08-02 22:11  何浩源  阅读(982)  评论(0编辑  收藏  举报
//一下两个链接最好自己保存下来,再上传到自己的博客园的“文件”选项中