第一版

package cache;

import java.util.HashMap;
import java.util.Map;

public class Cache1 {
	private Map<String, Object> map=new HashMap<String, Object>();

	private static Cache1 cache1=new Cache1();
	private Cache1(){

	}
	public static  Cache1 getInstanceCache1(){
		return cache1;
	}

	public void setObject(String key,Object value){
		map.put(key, value);
	}

	public Object getObject(String key){
		Object value=null;
		value=map.get(key);
		if (value==null) {
			value=getFromDB(key); // 从远程数据库获得
			map.put(key, value);
		}
		return value;
	}
	/**
	*仅仅只是模拟
	*/
	private Object getFromDB(String key) {
		return null;
	}
}

所谓的缓存,就是把经常用的数据存储到内存中,下次用的时候能很快的拿到。因而,上面的核心代码其实就是getObject。但是,得承认,上面的代码实在是太过简陋了。上面的测试程序很简单我就不写了。

第二版

之前第一版的缓存只是个实例代码,还算不上工具,因为它并没有对某一个"计算"操作做包装。我知道我说的很模糊,咱们看代码。

package cache;

public interface GetResutl {
	public Object get(Object o);
}

package cache;

public class ComputeSum implements GetResutl {

	@Override
	public Object get(Object o) {
		if (      !(o instanceof Integer)   ){
			throw new IllegalArgumentException (o+"is not Integer");

		}
		int n=(int) o;
		int result=0;
		for (int i = 1; i <= n; i++) {
			result+=i;
		}
		return result;
	}

}

package cache;

public class ComputeMultiply implements GetResutl {

	@Override
	public Object get(Object o) {
		if (      !(o instanceof Integer)   ){
			throw new IllegalArgumentException (o+"is not Integer");

		}
		int n=(int) o;
		int result=1;
		for (int i = 1; i <= n; i++) {
			result*=i;
		}
		return result;
	}

}

第一版的缓存系统,不能缓存某种操作的结果。

看了上面的连加,与连乘。

我们就大概知道新的缓存该是个什么样了。

package cache;

import java.util.HashMap;
import java.util.Map;

public  class Cache2 {
	private Map<String, Object> map=new HashMap<String, Object>();
	private GetResutl getResutl=null;

	public Cache2(GetResutl r){
		this.getResutl=r;
	}

	public void setObject(String key,Object value){
		map.put(key, value);
	}

	public Object getObject(String key){
		Object value=null;
		value=map.get(key);
		if (value==null) {
			value=getResutl.get(key);
			map.put(key, value);
		}
		return value;
	}
}

另外多嘴一句,Cahe2中有一个接口GetResut,之后再调用GetResult的某一个方法,这种设计似乎叫做策略模式。

第三版

如果ComputeMultiply中的get方法的运行需要花很长时间,同时我们也不着急要它计算的结果,第二版的计算是放在一个线程里的,这样效率不高。

我们试试CallAble。

package cache;

import java.util.concurrent.Callable;
import java.util.concurrent.FutureTask;

public class ComputeSum implements GetResutl {

	@Override
	public Object get(Object o) {

		final Long n=Long.valueOf((String) o);

		Callable<Long> c=new Callable<Long>() {
			@Override
			public Long call(){
				Long result=0L;
				for (int i = 1; i <= n; i++) {
					result+=i;
				}
				return result;
			}
		};

		return new FutureTask<>(c);
	}
}

连乘的写法与之类似,不再赘述。

再看cache的写法。

</pre><pre name="code" class="java">package cache;

import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;
import java.util.concurrent.FutureTask;

public  class Cache3 {
	private Map<String, FutureTask<Object>> map=new HashMap<String, FutureTask<Object> >();
	private GetResutl getResutl=null;

	public Cache3(GetResutl r){
		this.getResutl=r;
	}

	@SuppressWarnings("unchecked")
	public Object getObject(String key){
		FutureTask<Object> value=null;
		FutureTask<Object> ft=map.get(key);
		if (ft==null) {
			value= (FutureTask<Object>) getResutl.get(key);
			map.put(key, value);
			ft=value;
		}
		ft.run();
		Object result=null;
		try {
			result = ft.get();
		} catch (InterruptedException | ExecutionException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		return result;
	}

我们看看版本三的测试代码

public static void main(String[] args) {
		GetResutl getResutl=new ComputeSum();
		Cache3 c3=new Cache3(getResutl);

		long t1=0;
		long t2=0;

		t1=System.currentTimeMillis();
		System.out.println(c3.getObject(""+1234566));
		t2=System.currentTimeMillis();
		System.out.println(t2-t1 );

		t1=System.currentTimeMillis();
		System.out.println(c3.getObject(""+1234566));
		t2=System.currentTimeMillis();
		System.out.println(t2-t1 );

		t1=System.currentTimeMillis();
		System.out.println(c3.getObject(""+1234567));
		t2=System.currentTimeMillis();
		System.out.println(t2-t1 );

		t1=System.currentTimeMillis();
		System.out.println(c3.getObject(""+1234567));
		t2=System.currentTimeMillis();
		System.out.println(t2-t1 );

		t1=System.currentTimeMillis();
		System.out.println(c3.getObject(""+1234567));
		t2=System.currentTimeMillis();
		System.out.println(t2-t1 );
	}

输出:

762077221461

16

762077221461

0

762078456028

15

762078456028

0

762078456028

0

第四版

java并发编程中提到的一个例子。

import java.util.concurrent.Callable;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;
import java.util.concurrent.FutureTask;

interface Computable<K,V>{
    V compute(final K arg);
}

/**
 * 实现简单缓存系统
 * @author mzy
 *
 * @param <K> key
 * @param <V> value
 */
public class FutureCache<K,V> implements Computable<K,V>{
    private final ConcurrentHashMap<K, Future<V>> cache = new ConcurrentHashMap<K ,Future<V>>();
    private final Computable<K, V> c;
    public FutureCache(Computable<K, V> c) {
        this.c = c;
    }

    @Override
    public V compute(final K key) {
        while(true){
            Future<V> future = cache.get(key);
            if(future == null){
                Callable<V> eval = new Callable<V>() {
                    @Override
                    public V call() throws Exception { return c.compute(key); }
                };
                FutureTask<V> ftask = new FutureTask<V>(eval);
                //使用putIfAbsent原子操作避免有上面if(future == null)引起的相同值的缺陷
                future = cache.putIfAbsent(key, ftask);
                if(future == null) { future = ftask; ftask.run(); }
            }
            try {
                return future.get();
            } catch (InterruptedException e) {
                //出现中断异常应该从 cache中移除Future,防止缓存污染
                cache.remove(key,future);
            } catch (ExecutionException e) {
                //执行中的异常应当抛出,获得恰当处理
                throw new RuntimeException(e.getCause());
            }
        }
    }

}

测试程序:
public class Test {
    public static void main(String[] args) {
        final Computable<Integer, Integer> c = new Computable<Integer, Integer>() {
            @Override
            public Integer compute(Integer arg) {
                Integer sum = 0;
                for(Integer i=0;i<arg;i++){
                    sum+=i;
                }
                return sum;
            }
        };
        final Computable<Integer, Integer> cache = new FutureCache<Integer,Integer>(c);
        long start = System.currentTimeMillis();
        cache.compute(10000);
        long stop = System.currentTimeMillis();
        System.out.println(stop-start);
        start = System.currentTimeMillis();
        cache.compute(10000);
        stop = System.currentTimeMillis();
        System.out.println(stop-start);
        start = System.currentTimeMillis();
        cache.compute(10000);
        stop = System.currentTimeMillis();
        System.out.println(stop-start);
        start = System.currentTimeMillis();
        cache.compute(10000);
        stop = System.currentTimeMillis();
        System.out.println(stop-start);
    }
}

参考资料

http://my.oschina.net/ccdvote/blog/131876?p=1

java缓存系统的更多相关文章

  1. Java核心知识点学习----线程中如何创建锁和使用锁 Lock,设计一个缓存系统

    理论知识很枯燥,但这些都是基本功,学完可能会忘,但等用的时候,会发觉之前的学习是非常有意义的,学习线程就是这样子的. 1.如何创建锁? Lock lock = new ReentrantLock(); ...

  2. 缓存系统MemCached的Java客户端优化历程

    Memcached 是什么? Memcached是一种集中式Cache,支持分布式横向扩展.这里需要解释说明一下,很多开发者觉得Memcached是一种分布式缓存系统,但是其实Memcached服务端 ...

  3. Java核心知识点 --- 线程中如何创建锁和使用锁 Lock , 设计一个缓存系统

    理论知识很枯燥,但这些都是基本功,学完可能会忘,但等用的时候,会发觉之前的学习是非常有意义的,学习线程就是这样子的. 1.如何创建锁? Lock lock = new ReentrantLock(); ...

  4. JAVA 并发编程-读写锁之模拟缓存系统(十一)

    在多线程中,为了提高效率有些共享资源同意同一时候进行多个读的操作,但仅仅同意一个写的操作,比方一个文件,仅仅要其内容不变能够让多个线程同一时候读,不必做排他的锁定,排他的锁定仅仅有在写的时候须要,以保 ...

  5. [Java 缓存] Java Cache之 DCache的简单应用.

    前言 上次总结了下本地缓存Guava Cache的简单应用, 这次来继续说下项目中使用的DCache的简单使用. 这里分为几部分进行总结, 1)DCache介绍; 2)DCache配置及使用; 3)使 ...

  6. Memcache缓存系统构建一

    在如今这个高效率的社会中,怎样将这个高效率应用到自己的程序中,是一个值得追寻和值得探讨的问题.因为这个memcache能够很好的提高检索速度,提升用户体验,而且重要的是减少数据库的访问.这就大大的提高 ...

  7. 深入探讨在集群环境中使用 EhCache 缓存系统

    EhCache 缓存系统简介 EhCache 是一个纯 Java 的进程内缓存框架,具有快速.精干等特点,是 Hibernate 中默认的 CacheProvider. 下图是 EhCache 在应用 ...

  8. (转)java缓存技术,记录

    http://blog.csdn.net/madun/article/details/8569860 最近再ITEYE上看到关于讨论JAVA缓存技术的帖子比较多,自己不懂,所以上网大概搜了下,找到一篇 ...

  9. JAVA缓存技术

    介绍 JNotify:http://jnotify.sourceforge.net/,通过JNI技术,让Java代码可以实时的监控制定文件夹内文件的变动信息,支持Linux/Windows/MacOS ...

随机推荐

  1. android NDK的下载-文件太大

    需要FQ,建议使用VPN,下载前准备点时间配置网络环境.我的百度网盘好像有~~不过忘记地址了,改天共享,或者私聊我. 2015.4 Android 5.1 Android Studio https:/ ...

  2. Android自动打包工具aapt详解

    概念 在Android.mk中有LOCAL_AAPT_FLAGS配置项,在gradle中也有aaptOptions,那么aapt到底是干什么的呢? aapt即Android Asset Packagi ...

  3. Android性能提升之强引用、软引用、弱引用、虚引用使用

    转载请把头部出处链接和尾部二维码一起转载,本文出自逆流的鱼yuiop:http://blog.csdn.net/hejjunlin/article/details/52637333 背景:收到公众投稿 ...

  4. Web服务,XFire的一个例子

    Web服务优点 互操作性:实现不同系统间的相互调用(语言无关.平台无关) Web服务是什么 Web 服务是一类应用程序,是能够用编程的方法通过Web调用来实现某个功能的应用程序 Web服务的体系结构 ...

  5. 全文检索Lucene (1)

    Lucene是apache开源的一个全文检索框架,很是出名.今天先来分享一个类似于HelloWorld级别的使用. 工作流程 依赖 我们要想使用Lucene,那就得先引用人家的jar包了.下面列举一下 ...

  6. JAVA面向对象-----extends关键字

    继承使用extends关键字实现 1:发现学生是人,工人是人.显然属于is a 的关系,is a就是继承. 2:谁继承谁? 学生继承人,发现学生里的成员变量,姓名和年龄,人里边也都进行了定义.有重 复 ...

  7. everything of people’s life can changed in their twenties

    还记得三年前,独自背着行李,流浪远方,来到曾经只在地理课本上才熟悉的北国,带着好奇,带着期望,带着激动的心情,想感受毛爷爷当年霸气的北国风光,千里冰封的美丽,想知道北方的面条到底有多少种花样,想走进那 ...

  8. MySQL 视图技术

    以前也只是知道数据库中有视图这么个概念,但是没有去深究,今天正好有时间,就来总结一下吧. 视图的定义 视图就是从一个或多个表中,导出来的表,是一个虚拟存在的表.视图就像一个窗口(数据展示的窗口),通过 ...

  9. ubunut系统清理系统根目录下缓存文件夹.cache超大导致磁盘不足

    在使用中突然发现系统超慢,没有做什么特别的操作. 只好重启下电脑,重启后提示系统空间不足1G.挨个查看文件夹大小,没有发现问题,然后就用Ctrl + H显示隐藏文件夹后再继续逐个查看大小,发现.cac ...

  10. LM算法与非线性最小二乘问题

    摘录的一篇有关求解非线性最小二乘问题的算法--LM算法的文章,当中也加入了一些我个人在求解高精度最小二乘问题时候的一些感触: LM算法,全称为Levenberg-Marquard算法,它可用于解决非线 ...