缓存机制的实现

转自:http://www.iteye.com/topic/692103

原理:将Cache的所有位置都用双连表连接起来,当一个位置被命中之后,就将通过调整链表的指向,将该位置调整到链表头的位置,新加入的Cache直接加到链表头中。 

这样,在多次进行Cache操作后,最近被命中的,就会被向链表头方向移动,而没有命中的,而想链表后面移动,链表尾则表示最近最少使用的Cache。

当需要替换内容时候,链表的最后位置就是最少被命中的位置,我们只需要淘汰链表最后的部分即可。  帅!

import java.util.Hashtable;
/*
* 来源于:
http://www.iteye.com/topic/692103
*
*/
public class LRU {

private int cacheSize;
private Hashtable nodes;//缓存容器
private int currentSize;
private CacheNode first;//链表头
private CacheNode last;//链表尾

class CacheNode {
CacheNode prev;
//前一节点
CacheNode next;//后一节点
Object value;//
int key;//
CacheNode() {
}
}

public LRU(int i) {
currentSize
= 0;
cacheSize
= i;
nodes
= new Hashtable(i);//缓存容器
}

//获取缓存中对象
public Object get(int key) {
CacheNode node
= (CacheNode) nodes.get(key);

if (node != null) {
moveToHead(node);
System.out.println(
"get lurcache:" + key);
System.out.println(
"move lurcache:" + key + " to head");

return node.value;
}
else {
return null;
}
}

// 添加缓存
public void put(int key, Object value) {
CacheNode node
= (CacheNode) nodes.get(key);

if (node == null) {
System.out.println(
"insert into lurcache:" + key);
//缓存容器是否已经超过大小.
if (currentSize >= cacheSize) {
System.out.println(
"缓存不够了,需要移除最后一个 ");

if (last != null)//将最少使用的删除
{
System.out.println(
"========================");
nodes.remove(last.key);
}

removeLast();
}
else {

currentSize
++;
System.out.println(
"缓存大小为:" + currentSize);
}

node
= new CacheNode();
}
node.value
= value;
node.key
= key;
//将最新使用的节点放到链表头,表示最新使用的.
moveToHead(node);
nodes.put(key, node);
}

//将缓存删除
public Object remove(int key) {
CacheNode node
= (CacheNode) nodes.get(key);
if (node != null) {
if (node.prev != null) {
node.prev.next
= node.next;
}
if (node.next != null) {
node.next.prev
= node.prev;
}
if (last == node)
last
= node.prev;
if (first == node)
first
= node.next;
}
return node;
}

public void clear() {
first
= null;
last
= null;
}


//删除链表尾部节点,表示 删除最少使用的缓存对象
private void removeLast() {
System.out.println(
"移除中...");
//链表尾不为空,则将链表尾指向null. 删除连表尾(删除最少使用的缓存对象)
if (last != null) {

if (last.prev != null)
last.prev.next
= null;
else
first
= null;
last
= last.prev;
}
}

//移动到链表头,表示这个节点是最新使用过的
private void moveToHead(CacheNode node) {
if (node == first)
return;
if (node.prev != null)
node.prev.next
= node.next;
if (node.next != null)
node.next.prev
= node.prev;
if (last == node)
last
= node.prev;
if (first != null) {
node.next
= first;
first.prev
= node;
}
first
= node;
node.prev
= null;
if (last == null)
last
= first;
}

public void print_list()
{
CacheNode temp
= first;
while(temp != last)
{
System.out.print(temp.key
+ " ");
temp
= temp.next;
}
System.out.println(
"\n");
}


public static void main(String [] args)
{
LRU lrucache
= new LRU(10);
CacheNode node1
= lrucache.new CacheNode();
lrucache.put(
1, node1);
CacheNode node2
= lrucache.new CacheNode();
lrucache.put(
2, node2);
CacheNode node3
= lrucache.new CacheNode();
lrucache.put(
3, node3);
CacheNode node4
= lrucache.new CacheNode();
lrucache.put(
4, node4);
CacheNode node5
= lrucache.new CacheNode();
lrucache.put(
5, node5);
CacheNode node6
= lrucache.new CacheNode();
lrucache.put(
6, node6);
CacheNode node7
= lrucache.new CacheNode();
lrucache.put(
7, node7);
CacheNode node8
= lrucache.new CacheNode();
lrucache.put(
8, node8);
CacheNode node9
= lrucache.new CacheNode();
lrucache.put(
9, node9);
CacheNode node10
= lrucache.new CacheNode();
lrucache.put(
10, node10);
CacheNode node11
= lrucache.new CacheNode();
lrucache.put(
11, node11);

lrucache.print_list();

CacheNode node12
= lrucache.new CacheNode();
lrucache.put(
12, node12);

lrucache.print_list();

lrucache.get(
10);
lrucache.get(
10);
lrucache.print_list();
lrucache.get(
3);
lrucache.get(
3);
lrucache.print_list();
}
}

  最后加一个锁的并发访问吧,对于读锁与写锁的问题,加了写锁之后,就不能在添加读锁了,读锁之后还可以加写锁 。。

     没什么用,但是害怕丢了

import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;

public class LockTest1{
final
static ReadWriteLock rwlock = new ReentrantReadWriteLock();
final
static Lock readLock = rwlock.readLock();
final
static Lock writeLock = rwlock.writeLock();

public static void main(String[] args) throws InterruptedException {
System.
out.println("test readlock-writelock");
lockR2W();
System.
out.println("test writelock-readlock");
lockW2R();
System.
out.println("test readlock-readlock");
lockRR();
System.
out.println("test writelock-writelock");
lockWW();
}

private static void lockWW() {
writeLock.
lock();
System.
out.println("w lock.");
try {
if (writeLock.tryLock(1, TimeUnit.SECONDS))
System.
out.println("w lock.");
else
System.
out.println("when w lock, cannot w lock again.");
}
catch (InterruptedException e0) {
e0.printStackTrace();
}
try {
writeLock.unlock();
System.
out.println("w unlock.");
writeLock.unlock();
System.
out.println("w unlock.");
}
catch (Exception e) {
//ignore;
}
}

private static void lockRR() {
readLock.
lock();
System.
out.println("r lock.");
readLock.
lock();
System.
out.println("r lock.");
readLock.unlock();
System.
out.println("r unlock.");
readLock.unlock();
System.
out.println("r unlock.");
}

private static void lockW2R() throws InterruptedException {
writeLock.
lock();
System.
out.println("w lock.");
if (readLock.tryLock(1, TimeUnit.SECONDS))
System.
out.println("r lock.");
else
System.
out.println("when w lock, cannot r lock.");
writeLock.unlock();
System.
out.println("w unlock.");
readLock.unlock();
System.
out.println("r unlock.");
}

private static void lockR2W() {
readLock.
lock();
System.
out.println("r lock.");
try {
if (writeLock.tryLock(1, TimeUnit.SECONDS))
System.
out.println("w lock.");
else
System.
out.println("when r lock, cannot w lock.");
}
catch (InterruptedException e0) {
e0.printStackTrace();
}
try {
readLock.unlock();
System.
out.println("r unlock.");
writeLock.unlock();
System.
out.println("w unlock.");
}
catch (Exception e) {
//ignore;
}
}
}

  

posted @ 2011-08-18 21:42  wtx  阅读(878)  评论(0编辑  收藏  举报