jetbrick.collection.TimedSizeCache Maven / Gradle / Ivy
/**
* Copyright 2013-2016 Guoqiang Chen, Shanghai, China. All rights reserved.
*
* Author: Guoqiang Chen
* Email: [email protected]
* WebURL: https://github.com/subchen
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package jetbrick.collection;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
/**
* 简单具有超时检测的 cache 实现.
*
* @author Guoqiang Chen
*/
public final class TimedSizeCache {
private final ReentrantReadWriteLock lock = new ReentrantReadWriteLock();
private final Lock readLock = lock.readLock();
private final Lock writeLock = lock.writeLock();
protected Map cacheMap;
protected int maxSize; // max cache size, 0 = no limit
public TimedSizeCache(int maxSize) {
this.maxSize = maxSize;
this.cacheMap = new HashMap();
}
public int getMaxSize() {
return maxSize;
}
public void put(String key, Object object) {
put(key, object, 0);
}
public void put(String key, Object object, long timeout) {
writeLock.lock();
try {
CacheEntry entry = new CacheEntry(key, object, timeout);
if (isFull()) {
pruneCache();
if (isFull()) {
throw new IllegalStateException("exceed cache max size");
}
}
cacheMap.put(key, entry);
} finally {
writeLock.unlock();
}
}
public Object get(String key) {
readLock.lock();
try {
CacheEntry entry = cacheMap.get(key);
if (entry == null) {
return null;
}
if (entry.isExpired()) {
cacheMap.remove(key);
return null;
}
return entry.getObject();
} finally {
readLock.unlock();
}
}
public void prune() {
writeLock.lock();
try {
pruneCache();
} finally {
writeLock.unlock();
}
}
public boolean isEmpty() {
return size() == 0;
}
public boolean isFull() {
if (maxSize == 0) {
return false;
}
return cacheMap.size() >= maxSize;
}
public void remove(String key) {
writeLock.lock();
try {
cacheMap.remove(key);
} finally {
writeLock.unlock();
}
}
public void clear() {
writeLock.lock();
try {
cacheMap.clear();
} finally {
writeLock.unlock();
}
}
public int size() {
return cacheMap.size();
}
private void pruneCache() {
Iterator values = cacheMap.values().iterator();
while (values.hasNext()) {
CacheEntry entry = values.next();
if (entry.isExpired()) {
values.remove();
}
}
}
static class CacheEntry {
final String key;
final Object cachedObject;
long expiredTime; // time of expire
CacheEntry(String key, Object object, long ttl) {
this.key = key;
this.cachedObject = object;
this.expiredTime = (ttl == 0) ? 0 : System.currentTimeMillis() + ttl;
}
boolean isExpired() {
if (expiredTime == 0) {
return false;
}
return expiredTime < System.currentTimeMillis();
}
Object getObject() {
return cachedObject;
}
}
}