| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128 |
- //
- // A simple LRU cache
- //
- // Authors:
- // Miguel de Icaza ([email protected])
- // Andres G. Aragoneses ([email protected])
- //
- // Copyright 2010 Miguel de Icaza
- // Copyright 2013 7digital Media Ltd.
- //
- // Permission is hereby granted, free of charge, to any person obtaining a copy
- // of this software and associated documentation files (the "Software"), to deal
- // in the Software without restriction, including without limitation the rights
- // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- // copies of the Software, and to permit persons to whom the Software is
- // furnished to do so, subject to the following conditions:
- //
- // The above copyright notice and this permission notice shall be included in
- // all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- // THE SOFTWARE.
- //
- using System;
- using System.Collections.Generic;
- namespace System.Web.Configuration {
- internal class LruCache<TKey, TValue> {
- Dictionary<TKey, LinkedListNode <TValue>> dict;
- Dictionary<LinkedListNode<TValue>, TKey> revdict;
- LinkedList<TValue> list;
- int entryLimit;
- public LruCache (int entryLimit)
- {
- this.entryLimit = entryLimit;
- dict = new Dictionary<TKey, LinkedListNode<TValue>> ();
- revdict = new Dictionary<LinkedListNode<TValue>, TKey> ();
- list = new LinkedList<TValue> ();
- }
- //for debugging: public int Count { get { return dict.Count; } }
- void Evict ()
- {
- var last = list.Last;
- if (last == null)
- return;
- var key = revdict [last];
- dict.Remove (key);
- revdict.Remove (last);
- list.RemoveLast ();
- DisposeValue (last.Value);
- }
- public void Clear ()
- {
- foreach (var element in list) {
- DisposeValue (element);
- }
- dict.Clear ();
- revdict.Clear ();
- list.Clear ();
- }
- void DisposeValue (TValue value) {
- if (value is IDisposable) {
- ((IDisposable)value).Dispose ();
- }
- }
- public bool TryGetValue (TKey key, out TValue value)
- {
- LinkedListNode<TValue> node;
- if (dict.TryGetValue (key, out node)){
- list.Remove (node);
- list.AddFirst (node);
- value = node.Value;
- return true;
- }
- value = default (TValue);
- return false;
- }
- public void Add (TKey key, TValue value) {
- LinkedListNode<TValue> node;
- if (dict.TryGetValue (key, out node)){
- // If we already have a key, move it to the front
- list.Remove (node);
- list.AddFirst (node);
- // Remove the old value
- DisposeValue (node.Value);
- node.Value = value;
- return;
- }
- if (dict.Count >= entryLimit)
- Evict ();
- // Adding new node
- node = new LinkedListNode<TValue> (value);
- list.AddFirst (node);
- dict [key] = node;
- revdict [node] = key;
- }
- public override string ToString ()
- {
- return "LRUCache dict={0} revdict={1} list={2}";
- }
- }
- }
|