LRUMap源码分析

1.LRU算法

百度百科中对LRU算法的解释如下:

LRU是(Least Recently Used)近期最少使用算法。

其实就是把近期最少使用的内容淘汰出局。LRU是一种缓存算法,在空间有限的情况下,把使用频率最低的内容淘汰出缓存,让使用频率高的内容留在缓存里。

(2)在看项目框架的代码时,看到了LRUMap这个类,是apache-commons下的类,由于是第一次听说LRU算法,所以就决定研究一下它的源码。

(3)先来看一下例子:

 

 1 public class TestLRUMap {
 2     public static void main(String[] args) {
 3         LRUMap<Integer, Integer> map = new LRUMap<Integer, Integer>(3);
 4         map.put(1, 1);
 5         map.put(2, 2);
 6         map.put(3, 3);
 7         map.put(4, 4);
 8         System.out.println(map);
 9     }
10 }

 

输出:{2=2, 3=3, 4=4}

增加一行代码,再看结果:

 1 public class TestLRUMap {
 2     public static void main(String[] args) {
 3         LRUMap<Integer, Integer> map = new LRUMap<Integer, Integer>(3);
 4         map.put(1, 1);
 5         map.put(2, 2);
 6         map.put(3, 3);
 7         map.get(1);
 8         map.put(4, 4);
 9         System.out.println(map);
10     }
11 }

输出:{3=3, 1=1, 4=4}

两个结果为什么不一样呢?因为第二个例子,我们添加了一行代码:map.get(1);代表1元素是最新使用的,而相应的2就是最长时间没有使用的元素,而map的长度又限制为3,所以2就被删掉了。

(4)下面分析源码:

 首先看get方法:

1     public V get(final Object key) {
2         final LinkEntry<K, V> entry = getEntry(key);
3         if (entry == null) {
4             return null;
5         }
6         //主要看这个方法
7         moveToMRU(entry);
8         return entry.getValue();
9     }

moveToMRUst方法:

 1     protected void moveToMRU(final LinkEntry<K, V> entry) {
 2         if (entry.after != header) {
 3             modCount++;
 4             // remove
 5             if(entry.before == null) {
 6                 throw new IllegalStateException("Entry.before is null." +
 7                     " Please check that your keys are immutable, and that you have used synchronization properly." +
 8                     " If so, then please report this to dev@commons.apache.org as a bug.");
 9             }
10             //其实就是链表指正的改变,使得被get的元素排到链表中header前面
11             entry.before.after = entry.after;
12             entry.after.before = entry.before;
13             // add first
14             entry.after = header;
15             entry.before = header.before;
16             header.before.after = entry;
17             header.before = entry;
18         } else if (entry == header) {
19             throw new IllegalStateException("Can't move header to MRU" +
20                 " (please report this to dev@commons.apache.org)");
21         }
22     }

 接下来看put方法:

  1 public V put(final K key, final V value) {
  2         //如果key为null,则将其转换为new Object()
  3         final Object convertedKey = convertKey(key);
  4         //获得key的hashCode
  5         final int hashCode = hash(convertedKey);
  6         //根据hashCode得到其所在数组中的下标
  7         final int index = hashIndex(hashCode, data.length);
  8         //获得数组对应下标的HashEntry对象
  9         HashEntry<K, V> entry = data[index];
 10         while (entry != null) {
 11             //如果entry不为空,则判断是否已经有相同key的元素存在
 12             if (entry.hashCode == hashCode && isEqualKey(convertedKey, entry.key)) {
 13                 //判断是否有相同的key存在,如果有相同的key存在,则替换其值,但不会改变其在链表中的位置
 14                 final V oldValue = entry.getValue();
 15                 updateEntry(entry, value);
 16                 return oldValue;
 17             }
 18             entry = entry.next;
 19         }
 20         //如果entry为空,说明这个下标上还没有HashEntry对象
 21         addMapping(index, hashCode, key, value);
 22         return null;
 23     }
 24 
 25 
 26     protected void addMapping(final int hashIndex, final int hashCode, final K key, final V value) {
 27         //如果map已满,则执行删除到目前为止最长时间没有使用的元素
 28         if (isFull()) {
 29             //由于LRUMap每次添加元素的时候都放在链表中header的前面,所以,header后面的元素是删除的对象
 30             LinkEntry<K, V> reuse = header.after;
 31             boolean removeLRUEntry = false;
 32             if (scanUntilRemovable) {
 33                 while (reuse != header && reuse != null) {
 34                     if (removeLRU(reuse)) {
 35                         removeLRUEntry = true;
 36                         break;
 37                     }
 38                     reuse = reuse.after;
 39                 }
 40                 if (reuse == null) {
 41                     throw new IllegalStateException(
 42                         "Entry.after=null, header.after" + header.after + " header.before" + header.before +
 43                         " key=" + key + " value=" + value + " size=" + size + " maxSize=" + maxSize +
 44                         " Please check that your keys are immutable, and that you have used synchronization properly." +
 45                         " If so, then please report this to dev@commons.apache.org as a bug.");
 46                 }
 47             } else {
 48                 removeLRUEntry = removeLRU(reuse);
 49             }
 50 
 51             if (removeLRUEntry) {
 52                 if (reuse == null) {
 53                     throw new IllegalStateException(
 54                         "reuse=null, header.after=" + header.after + " header.before" + header.before +
 55                         " key=" + key + " value=" + value + " size=" + size + " maxSize=" + maxSize +
 56                         " Please check that your keys are immutable, and that you have used synchronization properly." +
 57                         " If so, then please report this to dev@commons.apache.org as a bug.");
 58                 }
 59                 //执行LRUMap的reuseMapping方法
 60                 reuseMapping(reuse, hashIndex, hashCode, key, value);
 61             } else {
 62                 super.addMapping(hashIndex, hashCode, key, value);
 63             }
 64         } else {
 65             super.addMapping(hashIndex, hashCode, key, value);
 66         }
 67     }
 68 
 69 
 70        protected void reuseMapping(final LinkEntry<K, V> entry, final int hashIndex, final int hashCode,
 71                                 final K key, final V value) {
 72         // find the entry before the entry specified in the hash table
 73         // remember that the parameters (except the first) refer to the new entry,
 74         // not the old one
 75         try {
 76             //获得删除元素在数组中的下标
 77             final int removeIndex = hashIndex(entry.hashCode, data.length);
 78             final HashEntry<K, V>[] tmp = data;  // may protect against some sync issues
 79             HashEntry<K, V> loop = tmp[removeIndex];
 80             HashEntry<K, V> previous = null;
 81             //loop的作用是防止map被其他线程修改了导致要删除的元素已经被其他线程给删了
 82             while (loop != entry && loop != null) {
 83                 previous = loop;
 84                 loop = loop.next;
 85             }
 86             if (loop == null) {
 87                 throw new IllegalStateException(
 88                     "Entry.next=null, data[removeIndex]=" + data[removeIndex] + " previous=" + previous +
 89                     " key=" + key + " value=" + value + " size=" + size + " maxSize=" + maxSize +
 90                     " Please check that your keys are immutable, and that you have used synchronization properly." +
 91                     " If so, then please report this to dev@commons.apache.org as a bug.");
 92             }
 93 
 94             // reuse the entry
 95             modCount++;
 96             removeEntry(entry, removeIndex, previous);
 97             reuseEntry(entry, hashIndex, hashCode, key, value);
 98             addEntry(entry, hashIndex);
 99         } catch (final NullPointerException ex) {
100             throw new IllegalStateException(
101                     "NPE, entry=" + entry + " entryIsHeader=" + (entry==header) +
102                     " key=" + key + " value=" + value + " size=" + size + " maxSize=" + maxSize +
103                     " Please check that your keys are immutable, and that you have used synchronization properly." +
104                     " If so, then please report this to dev@commons.apache.org as a bug.");
105         }
106     }

好了,通过源码分析,终于理解了LRUMap是怎么实现LRU算法的。

 

posted @ 2013-10-06 13:28  画水  阅读(977)  评论(0编辑  收藏  举报