longkeyclockcache.java

来自「RESIN 3.2 最新源码」· Java 代码 · 共 360 行

JAVA
360
字号
/* * Copyright (c) 1998-2008 Caucho Technology -- all rights reserved * * This file is part of Resin(R) Open Source * * Each copy or derived work must preserve the copyright notice and this * notice unmodified. * * Resin Open Source is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * Resin Open Source is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE, or any warranty * of NON-INFRINGEMENT.  See the GNU General Public License for more * details. * * You should have received a copy of the GNU General Public License * along with Resin Open Source; if not, write to the *   Free SoftwareFoundation, Inc. *   59 Temple Place, Suite 330 *   Boston, MA 02111-1307  USA * * @author Scott Ferguson */package com.caucho.util;import java.util.ArrayList;/** * Cache with a clock replacement policy. * * <p>Null keys are not allowed.  LruCache is synchronized. */public class LongKeyClockCache<E extends ClockCacheItem> {  // array containing the keys  private long []_keys;    // array containing the values  private E []_values;    // maximum allowed entries  private int _capacity;  // number of items in the cache  private int _size;  private int _mask;  private int _clock;    /**   * Create the clock cache with a specific capacity.   *   * @param initialCapacity minimum capacity of the cache   */  public LongKeyClockCache(int initialCapacity)  {    int capacity;    for (capacity = 16; capacity < 2 * initialCapacity; capacity *= 2) {    }    _keys = new long[capacity];    _values = (E []) new ClockCacheItem[capacity];    _mask = capacity - 1;    _capacity = initialCapacity;  }  /**   * Returns the current number of entries in the cache.   */  public int size()  {    return _size;  }  /**   * Clears the cache   */  public void clear()  {    ArrayList<E> listeners = null;    synchronized (this) {      for (int i = 0; i < _values.length; i++) {        E item = _values[i];        if (item != null) {	  if (listeners == null)	    listeners = new ArrayList<E>();	  listeners.add(item);        }                _values[i] = null;      }      _size = 0;    }    for (int i = listeners == null ? -1 : listeners.size() - 1; i >= 0; i--) {      E listener = listeners.get(i);      listener.removeEvent();    }  }  /**   * Get an item from the cache and make it most recently used.   *   * @param key key to lookup the item   * @return the matching object in the cache   */  public E get(long key)  {    int hash = getHash(key);    int count = _size + 1;    synchronized (this) {      for (; count > 0; count--) {        E item = _values[hash];        if (item == null)          return null;        if (_keys[hash] == key) {	  item.setUsed();          return item;        }        hash = (hash + 1) & _mask;      }    }    return null;  }  /**   * Puts a new item in the cache.  If the cache is full, remove the   * LRU item.   *   * @param key key to store data   * @param value value to be stored   *   * @return old value stored under the key   */  public E put(long key, E value)  {    freeSpace();    E item = putImpl(key, value);    // forced resizing if 3/4 full    if (3 * _values.length <= 4 * _size) {      synchronized (this) {	long []oldKeys = _keys;	E []oldValues = _values;		_keys = new long[2 * oldKeys.length];	_values = (E []) new Object[2 * oldValues.length];	_mask = _values.length - 1;	_size = 0;	for (int i = oldValues.length - 1; i >= 0; i--) {	  long oldKey = oldKeys[i];	  E oldValue = oldValues[i];	  if (oldValue != null)	    putImpl(oldKey, oldValue);	}      }    }    if (item != null)      item.removeEvent();    return item;  }  private E putImpl(long key, E value)  {    E item = null;    int hash = getHash(key);    int count = _size + 1;    synchronized (this) {      for (; count > 0; count--) {	item = _values[hash];	// No matching item, so create one	if (item == null) {	  _keys[hash] = key;	  _values[hash] = value;	  _size++;	  return null;	}	// matching item gets replaced	if (_keys[hash] == key) {	  item.setUsed();	  _values[hash] = value;	  return item;	}	hash = (hash + 1) & _mask;      }    }    throw new IllegalStateException();  }  private void freeSpace()  {    int i = _size - _capacity;    if (i > 16)      i = 16;    // remove LRU items until we're below capacity    while (i-- > 0 && removeItem()) {    }  }  /**   * Remove an unused item from the LRU   */  private boolean removeItem()  {    int length = _values.length;    int clock = _clock;    ClockCacheItem item = null;    for (int i = 0; i < length; i++) {      synchronized (this) {	item = _values[clock];	if (item != null && ! item.isUsed()) {	  _clock = clock;	  _values[clock] = null;	  _size--;	  refillEntries(clock);	  break;	}      }      if (item != null)	item.clearUsed();      clock = (clock + 1) % length;      item = null;    }    _clock = clock;    if (item != null) {      item.removeEvent();      return true;    }    else      return false;  }  /**   * Removes an item from the cache   *   * @param key the key to remove   *   * @return the value removed   */  public ClockCacheItem remove(long key)  {    int hash = getHash(key);    int count = _size + 1;    ClockCacheItem item = null;    synchronized (this) {      for (; count > 0; count--) {	item = _values[hash];	if (item == null)	  return null;	if (_keys[hash] == key) {	  _values[hash] = null;	  _size--; 	  refillEntries(hash);	  break;	}	hash = (hash + 1) & _mask;      }    }    if (count < 0)      throw new RuntimeException("internal cache error");    item.removeEvent();    return item;  }  /**   * Put the item in the best location available in the hash table.   */  private void refillEntries(int hash)  {    for (int count = _size; count >= 0; count--) {      hash = (hash + 1) & _mask;      if (_values[hash] == null)	return;      refillEntry(hash);    }  }    /**   * Put the item in the best location available in the hash table.   */  private void refillEntry(int baseHash)  {    long key = _keys[baseHash];    E value = _values[baseHash];        _values[baseHash] = null;          int hash = getHash(key);        for (int count = _size; count >= 0; count--) {      if (_values[hash] == null) {	_keys[hash] = key;	_values[hash] = value;	return;      }      hash = (hash + 1) & _mask;    }  }  /**   * Returns the key's hash   */  private int getHash(long key)  {    long hash = key;    hash = hash * 0x5deece66dl + 0xbl + (hash >>> 32) * 137;    return (int) hash & _mask;  }}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?