org.voovan.tools.bucket.LeakBucket Maven / Gradle / Ivy
package org.voovan.tools.bucket;
import org.voovan.tools.TEnv;
import org.voovan.tools.hashwheeltimer.HashWheelTask;
import java.util.concurrent.TimeoutException;
import java.util.concurrent.atomic.AtomicInteger;
/**
* 无锁漏桶
*
* @author: helyho
* DBase Framework.
* WebSite: https://github.com/helyho/DBase
* Licence: Apache v2 License
*/
public class LeakBucket extends Bucket {
private AtomicInteger atomicInteger = new AtomicInteger(0);
private long lastVisitTime = System.currentTimeMillis();
private int releaseTime = Integer.MAX_VALUE;
/**
* 令牌桶构造函数
* @param tokenSize 漏桶的初始数量
* @param interval 漏桶的新增周期, 每次触发将重置漏桶的数量, 单位: 毫秒
* @param releaseTime 漏桶失效并自动移除的时间
*/
public LeakBucket(int tokenSize, int interval, int releaseTime){
init(tokenSize, interval, releaseTime);
}
/**
* 令牌桶构造函数
* @param tokenSize 漏桶的初始数量
* @param interval 漏桶的新增周期, 每次触发将重置漏桶的数量, 单位: 毫秒
*/
public LeakBucket(int tokenSize, int interval){
init(tokenSize, interval, Integer.MAX_VALUE);
}
public void init(int tokenSize, int interval, int releaseTime){
this.releaseTime = releaseTime;
atomicInteger.set(tokenSize);
this.hashWheelTask = new HashWheelTask() {
@Override
public void run() {
if(System.currentTimeMillis() - lastVisitTime >= releaseTime){
this.cancel();
} else {
atomicInteger.set(tokenSize);
}
}
};
//重置漏桶的任务
Bucket.BUCKET_HASH_WHEEL_TIMER.addTask(this.hashWheelTask, interval);
}
public int getReleaseTime() {
return releaseTime;
}
public void setReleaseTime(int releaseTime) {
this.releaseTime = releaseTime;
}
/**
* 获取令牌, 立即返回
* @return true: 拿到令牌, false: 没有拿到令牌
*/
public boolean acquire() {
lastVisitTime = System.currentTimeMillis();
int value = atomicInteger.getAndUpdate((val) -> {
if(val <= 0){
return 0;
} else {
return val-1;
}
});
return value > 0 ;
}
/**
* 获取令牌, 带有时间等待
* @param timeout 等待时间
* @throws TimeoutException 超时异常
*/
public void acquire(int timeout) throws TimeoutException {
lastVisitTime = System.currentTimeMillis();
TEnv.wait(timeout, ()->!acquire());
}
}