首页 > 代码库 > 用Linkedhashmap的LRU特性及SoftReference软引用构建二级缓存

用Linkedhashmap的LRU特性及SoftReference软引用构建二级缓存

LRU: least recently used(近期最少使用算法)。LinkedHashMap构造函数可以指定其迭代顺序:LinkedHashMap(int initialCapacity, float loadFactor, boolean accessOrder) 设置accessOrder为true,则按照访问顺序迭代。当linkedhashmap调用put或者putall成功插入键值时,会调用removeEldestEntry方法,根据该方法的返回值决定是否删除最老对象(accessOrder为true后根据访问顺序迭代),为了保证map的size不超过某个值,可以重写removeEldestEntry方法,返回true,删除最老对象。

softreference软引用,通过例如new SoftReference<Object>(new Object())这样的方式来保留对一个object的软引用,在即将OOM的时候,GC会将softreference引用的对象回收。所以,在内存充足的时候,它的get()方法返回的是它引用的对象,在因为即将OOM导致GC回收之后,它的get方法返回的是null。

FirstCache.java:

public class FirstCache<K,V> extends LinkedHashMap<K,V>{    private static final long serialVersionUID = 1L;    private int MAX_SIZE = 100;    private SecondCache<K, V> secondCache = new SecondCache<K, V>();    public FirstCache(){        super();    }    public FirstCache(int max_size){        //利用linkedHashMap构造方法的accessOrder属性来构建LRU缓存        //accessOrder为true时,按照访问顺序排序,当accessOrder为false时,按照插入顺序排序        super(100,0.75f,true);        this.MAX_SIZE = max_size;    }    //当map调用put或者putall方法成功插入一个entry时,根据removeEldestEntry返回的bool值来确定是否删除least recently used对应的数据    //返回true删除  返回false保留    @Override    protected boolean removeEldestEntry(Entry<K,V> entry) {        if(size() >= MAX_SIZE){            //用softreference的特点来做二级缓存,softreference(软引用)只有在即将oom的时候 GC才会回收            secondCache.put(entry.getKey(), entry.getValue());            System.out.println("二级缓存容量" + secondCache.notEmptySize());            return true;        }        return false;    }}

SecondCache.java:

public class SecondCache<K,V> {    Map<K,SoftReference<V>> secondCacheMap = new HashMap<K, SoftReference<V>>();        public void put(K k,V v){        SoftReference<Object> o = new SoftReference<Object>(new Object());        secondCacheMap.put(k, new SoftReference<V>(v));    }        /**     * 将value为null的键值对删除,返回整个map中value不为null的数量     */    public int notEmptySize(){        int count = 0;        Iterator<Entry<K, SoftReference<V>>> iter = secondCacheMap.entrySet().iterator();        while(iter.hasNext()){            if(iter.next().getValue().get() == null)                iter.remove();            else                count++;        }        return count;    }}

测试方法:将虚拟机参数设置为-Xms64M -Xmx64M:

public class LRUCache {    public static final int MAX_SIZE = 20;    public static int count = 0;    public static void main(String[] args){        //一级缓存,利用linkedHashMap的LRU特性        FirstCache<Integer, Student> firstCache = new FirstCache<Integer, Student>(20);//        HashMap<Integer, Student> m = new HashMap<Integer, Student>();        while(true){            count++;            Student s = new Student();            //如果直接使用hashmap来存放,在count大约59的时候就抛出OOM了//            m.put(count, s);//            System.out.println(count);            firstCache.put(count, s);            if(count > MAX_SIZE){                System.out.println(firstCache.size());            }        }    }}

 

用Linkedhashmap的LRU特性及SoftReference软引用构建二级缓存