java缓存系统
第一版
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缓存系统的更多相关文章
- Java核心知识点学习----线程中如何创建锁和使用锁 Lock,设计一个缓存系统
理论知识很枯燥,但这些都是基本功,学完可能会忘,但等用的时候,会发觉之前的学习是非常有意义的,学习线程就是这样子的. 1.如何创建锁? Lock lock = new ReentrantLock(); ...
- 缓存系统MemCached的Java客户端优化历程
Memcached 是什么? Memcached是一种集中式Cache,支持分布式横向扩展.这里需要解释说明一下,很多开发者觉得Memcached是一种分布式缓存系统,但是其实Memcached服务端 ...
- Java核心知识点 --- 线程中如何创建锁和使用锁 Lock , 设计一个缓存系统
理论知识很枯燥,但这些都是基本功,学完可能会忘,但等用的时候,会发觉之前的学习是非常有意义的,学习线程就是这样子的. 1.如何创建锁? Lock lock = new ReentrantLock(); ...
- JAVA 并发编程-读写锁之模拟缓存系统(十一)
在多线程中,为了提高效率有些共享资源同意同一时候进行多个读的操作,但仅仅同意一个写的操作,比方一个文件,仅仅要其内容不变能够让多个线程同一时候读,不必做排他的锁定,排他的锁定仅仅有在写的时候须要,以保 ...
- [Java 缓存] Java Cache之 DCache的简单应用.
前言 上次总结了下本地缓存Guava Cache的简单应用, 这次来继续说下项目中使用的DCache的简单使用. 这里分为几部分进行总结, 1)DCache介绍; 2)DCache配置及使用; 3)使 ...
- Memcache缓存系统构建一
在如今这个高效率的社会中,怎样将这个高效率应用到自己的程序中,是一个值得追寻和值得探讨的问题.因为这个memcache能够很好的提高检索速度,提升用户体验,而且重要的是减少数据库的访问.这就大大的提高 ...
- 深入探讨在集群环境中使用 EhCache 缓存系统
EhCache 缓存系统简介 EhCache 是一个纯 Java 的进程内缓存框架,具有快速.精干等特点,是 Hibernate 中默认的 CacheProvider. 下图是 EhCache 在应用 ...
- (转)java缓存技术,记录
http://blog.csdn.net/madun/article/details/8569860 最近再ITEYE上看到关于讨论JAVA缓存技术的帖子比较多,自己不懂,所以上网大概搜了下,找到一篇 ...
- JAVA缓存技术
介绍 JNotify:http://jnotify.sourceforge.net/,通过JNI技术,让Java代码可以实时的监控制定文件夹内文件的变动信息,支持Linux/Windows/MacOS ...
随机推荐
- Android艺术开发探索第三章——View的事件体系(上)
Android艺术开发探索第三章----View的事件体系(上) 我们继续来看这本书,因为有点长,所以又分了上下,你在本片中将学习到 View基础知识 什么是View View的位置参数 Motion ...
- C++ ifstream,ofstream读写二进制文件
为什要吧数据存为二进制 这个嘛,是我个人习惯,一般,我们会把日志文件存为文本文件.数据文件存成二进制文件. 其实,我们接触的文件,比如图像.视频都是以二进制的形式存储的,要想查看这类数据,必须知道数据 ...
- Java安全管理器——SecurityManager
总的来说,Java安全应该包括两方面的内容,一是Java平台(即是Java运行环境)的安全性:二是Java语言开发的应用程序的安全性.由于我们不是Java本身语言的制定开发者,所以第一个安全性不需要我 ...
- java设计模式-----单例设计模式
设计模式是个很高深的东西,我也是略懂皮毛,下面让我用最简洁易懂的语言描述下单例设计模式吧. 一些人总结出来用来解决特定问题的固定的解决方案. 解决一个类在内存中只存在一个对象,想要保证对象的唯一. 1 ...
- 分布式缓存组件Hazelcast
Hazelcast是一个Java的开源分布式内存实现,它具有以下特性: 提供java.util.{Queue, Set, List, Map}的分布式实现 提供java.util.concurrent ...
- Android JavascriptBridge 详解(二)
原文出自:http://blog.csdn.net/sk719887916/article/details/47189607 Android开发目前现状来说,开发者大部分时间花在UI的屏幕适配上,使用 ...
- 14 Fragment的V4包的使用
activity_main.xml: <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android ...
- 4. React 属性和状态介绍
React 中的属性和状态初看之下可以互相替代,但是在 React 的设计哲学中两者有着截然不同的使用方式和使用场景. 属性的含义和用法 props = propert ...
- android程序崩溃后重启
有时候由于测试不充分或者程序潜在的问题而导致程序异常崩溃,这个是令人无法接受的,在android中怎样捕获程序的异常崩溃,然后进行一些必要的处理或重新启动 应用这个问题困恼了我很久,今天终于解决了该问 ...
- 读生产环境下go语言最佳实践有感
最近看了一篇关于go产品开发最佳实践的文章,go-in-procution.作者总结了他们在用go开发过程中的很多实际经验,我们很多其实也用到了,鉴于此,这里就简单的写写读后感,后续我也争取能将这篇文 ...