说实话在各种第三方工具和缓存机制的成熟方案太多了,使用Java原生的ConcurrentHashMap意义实在不大,但是有时候自己想偷懒就用个简单的缓存ConcurrentHashMap还是能用的上的,首先我们要了解ConcurrentHashMap是什么?
ConcurrentHashMap的简要总结:
1、public V get(Object key)不涉及到锁,也就是说获得对象时没有使用锁;
2、put、remove方法要使用锁,但并不一定有锁争用,原因在于ConcurrentHashMap将缓存的变量分到多个Segment,每个Segment上有一个锁,只要多个线程访问的不是一个Segment就没有锁争用,就没有堵塞,各线程用各自的锁,ConcurrentHashMap缺省情况下生成16个Segment,也就是允许16个线程并发的更新而尽量没有锁争用;
3、Iterator对象的使用,不一定是和其它更新线程同步,获得的对象可能是更新前的对象,ConcurrentHashMap允许一边更新、一边遍历,也就是说在Iterator对象遍历的时候,ConcurrentHashMap也可以进行remove,put操作,且遍历的数据会随着remove,put操作产出变化,所以希望遍历到当前全部数据的话,要么以ConcurrentHashMap变量为锁进行同步(synchronized该变量),要么使用CopiedIterator包装iterator,使其拷贝当前集合的全部数据,但是这样生成的iterator不可以进行remove操作。
Hashtable和ConcurrentHashMap的不同点:
1、Hashtable对get,put,remove都使用了同步操作,它的同步级别是正对Hashtable来进行同步的,也就是说如果有线程正在遍历集合,其他的线程就暂时不能使用该集合了,这样无疑就很容易对性能和吞吐量造成影响,从而形成单点。而ConcurrentHashMap则不同,它只对put,remove操作使用了同步操作,get操作并不影响,详情请看以上第1,2点,当前ConcurrentHashMap这样的做法对一些线程要求很严格的程序来说,还是有所欠缺的,对应这样的程序来说,如果不考虑性能和吞吐量问题的话,个人觉得使用Hashtable还是比较合适的;
2、Hashtable在使用iterator遍历的时候,如果其他线程,包括本线程对Hashtable进行了put,remove等更新操作的话,就会抛出ConcurrentModificationException异常,但如果使用ConcurrentHashMap的话,就不用考虑这方面的问题了,详情请看以上第3点;
我们用ConcurrentHashMap实现一个简单的缓存,功能有set、get和设置缓存值的时间,并且能定时清除缓存。
代码参考:https://blog.csdn.net/qq_35030994/article/details/80871279
package cn.yan.study.utils.cache;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.util.*;
import java.util.concurrent.*;
/**
* Created
* User study
* Date 2018/6/26
* Time 20:32
*/
public class ConcurrentHashMapCacheUtils {
private static Logger LOGGER = LoggerFactory.getLogger(ConcurrentHashMapCacheUtils.class);
/**
* 缓存最大个数
*/
private static final Integer CACHE_MAX_NUMBER = 100;
/**
* 当前缓存个数
*/
private static Integer CURRENT_SIZE = 0;
/**
* 时间一分钟
*/
static Long ONE_MINUTE = 1 * 60 * 1000L;
/**
* 缓存对象
*/
private static final Map CACHE_OBJECT_MAP = new ConcurrentHashMap<>();
/**
* 这个记录了缓存使用的最后一次的记录,最近使用的在最前面
*/
private static final List CACHE_USE_LOG_LIST = new LinkedList<>();
/**
* 清理过期缓存是否在运行
*/
private static Boolean CLEAN_THREAD_IS_RUN = false;
/**
* 清理线程的
*/
private static ExecutorService executor = Executors.newSingleThreadExecutor();
/**
* 设置缓存
*/
public static void setCache(String cacheKey, Object cacheValue, long cacheTime) {
Long ttlTime = null;
if (cacheTime <= 0L) {
if (cacheTime == -1L) {
ttlTime = -1L;
} else {
return;
}
}
checkSize();
saveCacheUseLog(cacheKey);
CURRENT_SIZE = CURRENT_SIZE + 1;
if (ttlTime == null) {
ttlTime = System.currentTimeMillis() + cacheTime;
}
CacheObj cacheObj = new CacheObj(cacheValue, ttlTime);
CACHE_OBJECT_MAP.put(cacheKey, cacheObj);
LOGGER.info("have set key :" + cacheKey);
}
/**
* 设置缓存
*/
public static void setCache(String cacheKey, Object cacheValue) {
setCache(cacheKey, cacheValue, -1L);
}
/**
* 获取缓存
*/
public static Object getCache(String cacheKey) {
startCleanThread();
if (checkCache(cacheKey)) {
saveCacheUseLog(cacheKey);
return CACHE_OBJECT_MAP.get(cacheKey).getCacheValue();
}
return null;
}
public static boolean isExist(String cacheKey) {
return checkCache(cacheKey);
}
/**
* 删除所有缓存
*/
public static void clear() {
LOGGER.info("have clean all key !");
CACHE_OBJECT_MAP.clear();
CURRENT_SIZE = 0;
}
/**
* 删除某个缓存
*/
public static void deleteCache(String cacheKey) {
Object cacheValue = CACHE_OBJECT_MAP.remove(cacheKey);
if (cacheValue != null) {
LOGGER.info("have delete key :" + cacheKey);
CURRENT_SIZE = CURRENT_SIZE - 1;
}
}
/**
* 判断缓存在不在,过没过期
*/
private static boolean checkCache(String cacheKey) {
CacheObj cacheObj = CACHE_OBJECT_MAP.get(cacheKey);
if (cacheObj == null) {
return false;
}
if (cacheObj.getTtlTime() == -1L) {
return true;
}
if (cacheObj.getTtlTime() < System.currentTimeMillis()) {
deleteCache(cacheKey);
return false;
}
return true;
}
/**
* 删除最近最久未使用的缓存
*/
private static void deleteLRU() {
LOGGER.info("delete Least recently used run!");
String cacheKey = CACHE_USE_LOG_LIST.remove(CACHE_USE_LOG_LIST.size() - 1);
deleteCache(cacheKey);
}
/**
* 删除过期的缓存
*/
static void deleteTimeOut() {
LOGGER.info("delete time out run!");
List deleteKeyList = new LinkedList<>();
for(Map.Entry entry : CACHE_OBJECT_MAP.entrySet()) {
if (entry.getValue().getTtlTime() < System.currentTimeMillis() && entry.getValue().getTtlTime() != -1L) {
deleteKeyList.add(entry.getKey());
}
}
for (String deleteKey : deleteKeyList) {
deleteCache(deleteKey);
}
LOGGER.info("delete cache count is :" + deleteKeyList.size());
}
/**
* 检查大小
* 当当前大小如果已经达到最大大小
* 首先删除过期缓存,如果过期缓存删除过后还是达到最大缓存数目
* 删除最久未使用缓存
*/
private static void checkSize() {
if (CURRENT_SIZE >= CACHE_MAX_NUMBER) {
deleteTimeOut();
}
if (CURRENT_SIZE >= CACHE_MAX_NUMBER) {
deleteLRU();
}
}
/**
* 保存缓存的使用记录
*/
private static synchronized void saveCacheUseLog(String cacheKey) {
CACHE_USE_LOG_LIST.remove(cacheKey);
CACHE_USE_LOG_LIST.add(0,cacheKey);
}
/**
* 设置清理线程的运行状态为正在运行
*/
static void setCleanThreadRun() {
CLEAN_THREAD_IS_RUN = true;
}
/**
* 开启清理过期缓存的线程
*/
private static void startCleanThread() {
if (!CLEAN_THREAD_IS_RUN) {
// new Thread(new CleanTimeOutThread()).run();
executor.submit(new CleanTimeOutThread());
}
}
public static void showUtilsInfo() {
System.out.println("clean time out cache is run :" + CLEAN_THREAD_IS_RUN);
System.out.println("cache max count is :" + CACHE_MAX_NUMBER);
System.out.println("cache current count is :" + CURRENT_SIZE);
System.out.println("cache object map is :" + CACHE_OBJECT_MAP.toString());
System.out.println("cache use log list is :" + CACHE_USE_LOG_LIST.toString());
}
public static void main(String[] args) {
for (int i = 0; i < 100; i++) {
try {
Thread.sleep(2 * 1000L);
} catch (InterruptedException e) {
e.printStackTrace();
}
ConcurrentHashMapCacheUtils.setCache("my_cache_key_" + i, i, 60*1000);
}
for (int i = 0; i < 100; i++) {
if (i > 10) {
ConcurrentHashMapCacheUtils.getCache("test");
}
try {
Thread.sleep(2 * 1000L);
} catch (InterruptedException e) {
e.printStackTrace();
}
ConcurrentHashMapCacheUtils.showUtilsInfo();
}
}
}
class CacheObj {
/**
* 缓存对象
*/
private Object CacheValue;
/**
* 缓存过期时间
*/
private Long ttlTime;
CacheObj(Object cacheValue, Long ttlTime) {
CacheValue = cacheValue;
this.ttlTime = ttlTime;
}
Object getCacheValue() {
return CacheValue;
}
Long getTtlTime() {
return ttlTime;
}
@Override
public String toString() {
return "CacheObj{" +
"CacheValue=" + CacheValue +
", ttlTime=" + ttlTime +
'}';
}
}
/**
* 每一分钟清理一次过期缓存
*/
class CleanTimeOutThread implements Runnable{
@Override
public void run() {
ConcurrentHashMapCacheUtils.setCleanThreadRun();
while (true) {
System.out.println("clean thread run ");
ConcurrentHashMapCacheUtils.deleteTimeOut();
try {
Thread.sleep(ConcurrentHashMapCacheUtils.ONE_MINUTE);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}