/* * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You under the Apache License, Version 2.0 * (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ using System; using System.Collections.Generic; namespace Mono.Utilities { public class LRUCache { [ThreadStatic] static LRUCache deflt; public static LRUCache Default { get { return deflt != null ? deflt : (deflt = new LRUCache (5)); } } int capacity; LinkedList> list; Dictionary>> lookup; LinkedListNode> openNode; public LRUCache (int capacity) { this.capacity = capacity; this.list = new LinkedList>(); this.lookup = new Dictionary>> (capacity + 1); this.openNode = new LinkedListNode>(new ListValueEntry (default(TKey), default(TValue))); } public void Put (TKey key, TValue value) { if (Get(key) == null) { this.openNode.Value.ItemKey = key; this.openNode.Value.ItemValue = value; this.list.AddFirst (this.openNode); this.lookup.Add (key, this.openNode); if (this.list.Count > this.capacity) { // last node is to be removed and saved for the next addition to the cache this.openNode = this.list.Last; // remove from list & dictionary this.list.RemoveLast(); this.lookup.Remove(this.openNode.Value.ItemKey); } else { // still filling the cache, create a new open node for the next time this.openNode = new LinkedListNode>(new ListValueEntry(default(TKey), default(TValue))); } } } public TValue Get (TKey key) { LinkedListNode> node = null; if (!this.lookup.TryGetValue (key, out node)) return default (TValue); this.list.Remove (node); this.list.AddFirst (node); return node.Value.ItemValue; } class ListValueEntry where K : TKey where V : TValue { internal V ItemValue; internal K ItemKey; internal ListValueEntry(K key, V value) { this.ItemKey = key; this.ItemValue = value; } } } }