zoukankan      html  css  js  c++  java
  • 缓存机制的实现

    转自: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;
    }
    }
    }

      

  • 相关阅读:
    Typecho
    iOS开发
    搭建自己的Git服务器
    Markdown
    微信使用篇
    bzoj2406 矩阵
    POI2010 Bridges
    上下界网络流总结
    TopCoder SRM 582 Div 1
    port命令记录
  • 原文地址:https://www.cnblogs.com/hitwtx/p/2145070.html
Copyright © 2011-2022 走看看