Guava包学习-Cache
这段时间用到了ehcache和memcache,memcache只用来配置在tomcat中做负载均衡过程中的session共享,然后ehcache用来存放需要的程序中缓存。
Guava中的Cache和ehcache其实差不多,只不过Guava并不会对数据进行持久化落盘这种操作。那其实和Map就比较相似了,你放一下static的map好像也可以办到,只不过Guava的cache支持设置最大占用的内存,已经命中统计信息之类的东西。
这里不从怎么创建一个cache,怎么去添加删除开始学习,我们先看cache下面的class文件的名称找到其中一个枚举:

里面有一个叫做RemovalCause的枚举,我们看下它的内容:
package com.google.common.cache;
@GwtCompatible
public enum RemovalCause {
/**
* The entry was manually removed by the user. This can result from the user invoking
* {@link Cache#invalidate}, {@link Cache#invalidateAll(Iterable)}, {@link Cache#invalidateAll()},
* {@link Map#remove}, {@link ConcurrentMap#remove}, or {@link Iterator#remove}.
*/--------------------------移除动作是用户自己去做的
EXPLICIT {
@Override
boolean wasEvicted() {
return false;
}
}, /**
* The entry itself was not actually removed, but its value was replaced by the user. This can
* result from the user invoking {@link Cache#put}, {@link LoadingCache#refresh}, {@link Map#put},
* {@link Map#putAll}, {@link ConcurrentMap#replace(Object, Object)}, or
* {@link ConcurrentMap#replace(Object, Object, Object)}.
*/---------------------------移除动作是用户的值覆盖了现有的值
REPLACED {
@Override
boolean wasEvicted() {
return false;
}
}, /**
* The entry was removed automatically because its key or value was garbage-collected. This
* can occur when using {@link CacheBuilder#weakKeys}, {@link CacheBuilder#weakValues}, or
* {@link CacheBuilder#softValues}.
*/------------------------移除的动作是因为垃圾回收给回收掉了,但是前提是在创建的时候声明是软引用或者弱引用的key、values
COLLECTED {
@Override
boolean wasEvicted() {
return true;
}
}, /**
* The entry's expiration timestamp has passed. This can occur when using
* {@link CacheBuilder#expireAfterWrite} or {@link CacheBuilder#expireAfterAccess}.
*/---------------因为时间够了自动去除了,前提是开启了超时时间设置
EXPIRED {
@Override
boolean wasEvicted() {
return true;
}
}, /**
* The entry was evicted due to size constraints. This can occur when using
* {@link CacheBuilder#maximumSize} or {@link CacheBuilder#maximumWeight}.
*/----------------因为超过了size满了,前提是你开启了最大内存占用或者最大kv量限制
SIZE {
@Override
boolean wasEvicted() {
return true;
}
}; /**
* Returns {@code true} if there was an automatic removal due to eviction (the cause is neither
* {@link #EXPLICIT} nor {@link #REPLACED}).
*/------如果不是枚举一和二的话,返回是否其他操作把它给移除掉了
abstract boolean wasEvicted();
}
从这个k,v从缓存中消失的事件来看,其实Guava的cache就是会有缓存大小上限和时间上限的限制,同时会有这些移除策略将缓存信息移除。
所以如果应用需要这种场景:
1、需要缓存,因为可以提高性能,可以有命中key的情况,并且我不需要持久化,应用重启重新来即可;
2、JVM中有足够的缓存给它用(其实这就是JVM提供弱引用和软引用的用处了)
3、对于使用Map来缓存的场景不足以满足统计信息和限制大小的情况
4、不想外接Ehcache和memcache的配置和应用,否则服务器迁移还得安装那些玩意儿,不如直接在JVM的堆里面搞
LoadingCache<String, String> cache = CacheBuilder.newBuilder()
.expireAfterWrite(1000, TimeUnit.MILLISECONDS)--------在写入或者更新1000毫秒后失效
.expireAfterAccess(200, TimeUnit.MILLISECONDS)--------在访问1000毫秒后失效
.concurrencyLevel(8).initialCapacity(100)--------这个地方有点像分段锁提高性能的ConcurrentHashMap,大小100,level 8 分13段
.maximumSize(100)---最多放100个key
.weakKeys().weakValues()---使用弱引用
.build(new CacheLoader<String, String>() {
@Override
public String load(String key) throws Exception {
return "hah";
}
});
     cache.put("hahah", "heiheihei");
        cache.put("hahah1", "heiheihei1");
        cache.put("hahah2", "heiheihei2");
        cache.put("hahah3", "heiheihei3");
        cache.put("hahah4", "heiheihei4");
        System.out.println(cache.asMap().toString());
        ConcurrentMap<String, String> map = cache.asMap();
        map.remove("hahah");
        System.out.println(map.toString());
        System.out.println(cache.asMap().toString());
        System.out.println(cache.get("hahah1"));
这里暂时不测试那些超时啊 超出大小限制啊之类的东西,可以sleep或者for循环一下去体验一下,都比较简单。至于还有其他的一些带callback获得get的或者refresh或者cleanup之类的都在使用的时候看一下它的注释应该就可以理解了。
这样子系统就可以使用这个缓存了,而不需要在以来外围的ehcache和memcache了。
try {
            System.out.println(cache.get("hahah1",new Callable<String>() {
                @Override
                public String call() throws Exception {
                    //这里要做的其实是如果你没有在缓存中命中这个key,那么你要怎么做?
                    return null;
                }
            }));
        } catch (ExecutionException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
我们发现去缓存获取key的情况之后可以写一个回掉方法,这对于需要前端进入缓存,发现未命中再透到后端去比较好用。
比如目前我们项目中的API,前端有redis集群,然后查询来了之后会拿key去redis中进行查询,如果redis中不存在,那么程序透到DB里面去查询,并且将查询到的结果放到redis中。
那么Guava的cache在你无法命中该key之后,你也可以通过自定义回调方法来进行类似的操作。
所谓的回掉方法,其实就一个new一个接口,然后接口中会有方法必须去实现~
查看Cache状态:
CacheStats cs = cache.stats();
System.out.println(cs.hitCount());
System.out.println(cs.hitRate());
Guava包学习-Cache的更多相关文章
- Guava包学习---Lists
		Guava包是我最近项目中同事推荐使用的,是google推出的库.里面的功能非常多,包括了集合.缓存.原生类型支持.并发库.通用注解.字符串处理.IO等.我们项目中使用到了guava依赖,但是实际上只 ... 
- Guava包学习--EventBus
		之前没用过这个EventBus,然后看了一下EventBus的源码也没看明白,(-__-)b.反正大概就是弄一个优雅的方式实现了观察者模式吧.慢慢深入学习一下. 观察者模式其实就是生产者消费者的一个变 ... 
- Guava包学习---Maps
		Maps包方法列表: 还是泛型创建Map: public static <K, V> HashMap<K, V> newHashMap() { return new HashM ... 
- Guava包学习--Hash
		我们HashMap会有一个rehash的过程,为什么呢?因为java内建的散列码被限制为32位,而且没有分离散列算法和所作用的数据,所以替代算法比较难做.我们使用HashMap的时候它自身有一个reh ... 
- Guava包学习---I/O
		Guava的I/O平时使用不太多,目前项目原因导致基本上只有在自己写一些文本处理小工具才用得到.但是I/O始终是程序猿最常遇到的需求和面试必问的知识点之一.同时Guava的I/O主要面向是时JDK5和 ... 
- Guava包学习---Bimap
		Bimap也是Guava中提供的新集合类,别名叫做双向map,就是key->value,value->key,也就是你可以通过key定位value,也可以用value定位key. 这个场景 ... 
- Guava包学习-Multimap
		它和上一章的MultiSet的继承结果很相似,只不过在上层的接口是Multimap不是Multiset. Multimap的特点其实就是可以包含有几个重复Key的value,你可以put进入多个不同v ... 
- Guava包学习---Sets
		Sets包的内容和上一篇中的Lists没有什么大的区别,里面有些细节可以看一下: 开始的创建newHashSet()的各个重载方法.newConcurrentHashSet()的重载方法.newTre ... 
- Guava包学习--Table
		Table,顾名思义,就好像HTML中的Table元素一样,其实就是行+列去确定的值,更准确的比喻其实就是一个二维矩阵. 其实它就是通过行+列两个key去找到一个value,然后它又containsv ... 
随机推荐
- jquery里判断数组内是否包含了指定的值或元素的方法
			本文讲的是在jquery里,如何判断一个数组里是否包含了指定的值,变量,或其它对象元素的方法. 在jquery里,我们可以用$.inArray来判断一个数组里是否包含了指定的值或其它对象元素,来看一个 ... 
- Office 卸载问题(安装包的语言不受系统支持)
			本人系统Win7 这个问题搞了一下午.各种网站找解决办法.下载下来的都是一些垃圾软件. Win7以上调成兼容模式运行理论可行. 放上微软的解决方法: * 彻底卸载Office 2003: http:/ ... 
- C# DateTime类型和sqlserver DateTime精度不同
			在最近的项目中, 有个关于时间的功能.一个请假的时间.前端选择的时候只有日期.所以比如请一天假就是选2017-8-15和2017-8-15,这样算请这一天的假.但是后台存入数据库时我不能就存2017- ... 
- java核心技术-NIO
			1.reactor(反应器)模式 使用单线程模拟多线程,提高资源利用率和程序的效率,增加系统吞吐量.下面例子比较形象的说明了什么是反应器模式: 一个老板经营一个饭店, 传统模式 - 来一个客人安排一个 ... 
- JSON 解析的可抛弃
			先看例子, json文件中有些元素不是我们想要的,在反序列化时可以当它们不存在,下面例子抛弃了 aaa.ccc这两节. package main import ( "encoding ... 
- HDFS学习
			HDFS体系结构 HDFS采用了主从(Master/Slave)结构模型,一个HDFS集群包括一个名称节点(NameNode)和若干个数据节点(DataNode)(如图所示).名称节点作为中心服务器, ... 
- maven 程序包com.sun.image.codec.jpeg
			在 Pom.xml 增加 <build> <plugins> <plugin> <artifactId>maven-compiler-plugin< ... 
- K:顺序表和链表的比较
			顺序表和链表是线性表的两种基本实现形式(链表还有多种变化形式),对于这两种实现方式,没有一种方法可以称是最好的,他们各自有着各自的特点和优缺点,适用于不同的应用场景. 与顺序表相比,链表较为灵活, ... 
- javascript之url转义escape()、encodeURI()和decodeURI(),ifram父子传参参数有中文时出现乱码
			ifram父子传参参数有中文时出现乱码,可先在父级页面用encodeURI转义,在到子页面用进行decodeURI()解码 我们可以知道:escape()除了 ASCII 字母.数字和特定的符号外,对 ... 
- android 获取http请求json数据
			package com.my.gethttpjsondata; import java.io.BufferedReader;import java.io.ByteArrayOutputStream;i ... 
