java实现时间轮算法

Main

demo测试代码

public class Main {static int inCount = 0;static int runCount = 0;public static void main(String[] args) {CountDownLatch countDownLatch = new CountDownLatch(1000);Timer timer = new Timer();for(int i=1;i<=1000;i++){TimerTask timerTask = new TimerTask(i,()->{countDownLatch.countDown();int index = addRun();System.out.println(index+"----------执行");});timer.addTask(timerTask);System.out.println(i+"++++++++++加入");inCount++;}TimerTask timerTask = new TimerTask(5000,()->{countDownLatch.countDown();int index = addRun();System.out.println(index+"----------执行");});timer.addTask(timerTask);try {countDownLatch.await();System.out.println("inCount" + inCount);System.out.println("runCount" + runCount);} catch (Exception e){e.printStackTrace();}}public synchronized static int addRun(){runCount++;return runCount;}}

Timer

时间轮定时器

*** 定时器*/
public class Timer {/*** 底层时间轮*/private TimeWheel timeWheel;/*** 一个Timer只有一个delayQueue*/private DelayQueue<TimerTaskList> delayQueue = new DelayQueue<>();/*** 过期任务执行线程*/private ExecutorService workerThreadPool;/*** 轮询delayQueue获取过期任务线程*/private ExecutorService bossThreadPool;/*** 构造函数*/public Timer() {timeWheel = new TimeWheel(1, 20, System.currentTimeMillis(), delayQueue);workerThreadPool = Executors.newFixedThreadPool(100);bossThreadPool = Executors.newFixedThreadPool(1);//20ms获取一次过期任务bossThreadPool.submit(() -> {while (true) {this.advanceClock(20);}});}/*** 添加任务*/public void addTask(TimerTask timerTask) {//添加失败任务直接执行if (!timeWheel.addTask(timerTask)) {workerThreadPool.submit(timerTask.getTask());}}/*** 获取过期任务*/private void advanceClock(long timeout) {try {TimerTaskList timerTaskList = delayQueue.poll(timeout, TimeUnit.MILLISECONDS);if (timerTaskList != null) {//推进时间timeWheel.advanceClock(timerTaskList.getExpiration());//执行过期任务(包含降级操作)timerTaskList.flush(this::addTask);}} catch (Exception e) {e.printStackTrace();}}
}

TimeWheel

时间轮

/*** 时间轮*/
public class TimeWheel {/*** 一个时间槽的范围*/private long tickMs;/*** 时间轮大小*/private int wheelSize;/*** 时间跨度*/private long interval;/*** 时间槽*/private TimerTaskList[] timerTaskLists;/*** 当前时间*/private long currentTime;/*** 上层时间轮*/private volatile TimeWheel overflowWheel;/*** 一个Timer只有一个delayQueue*/private DelayQueue<TimerTaskList> delayQueue;public TimeWheel(long tickMs, int wheelSize, long currentTime, DelayQueue<TimerTaskList> delayQueue) {this.currentTime = currentTime;this.tickMs = tickMs;this.wheelSize = wheelSize;this.interval = tickMs * wheelSize;this.timerTaskLists = new TimerTaskList[wheelSize];//currentTime为tickMs的整数倍 这里做取整操作this.currentTime = currentTime - (currentTime % tickMs);this.delayQueue = delayQueue;for (int i = 0; i < wheelSize; i++) {timerTaskLists[i] = new TimerTaskList();}}/*** 创建或者获取上层时间轮*/private TimeWheel getOverflowWheel() {if (overflowWheel == null) {synchronized (this) {if (overflowWheel == null) {overflowWheel = new TimeWheel(interval, wheelSize, currentTime, delayQueue);}}}return overflowWheel;}/*** 添加任务到时间轮*/public boolean addTask(TimerTask timerTask) {long expiration = timerTask.getDelayMs();//过期任务直接执行if (expiration < currentTime + tickMs) {return false;} else if (expiration < currentTime + interval) {//当前时间轮可以容纳该任务 加入时间槽Long virtualId = expiration / tickMs;int index = (int) (virtualId % wheelSize);System.out.println("tickMs:" + tickMs + "------index:" + index + "------expiration:" + expiration);TimerTaskList timerTaskList = timerTaskLists[index];timerTaskList.addTask(timerTask);if (timerTaskList.setExpiration(virtualId * tickMs)) {//添加到delayQueue中delayQueue.offer(timerTaskList);}} else {//放到上一层的时间轮TimeWheel timeWheel = getOverflowWheel();timeWheel.addTask(timerTask);}return true;}/*** 推进时间*/public void advanceClock(long timestamp) {if (timestamp >= currentTime + tickMs) {currentTime = timestamp - (timestamp % tickMs);if (overflowWheel != null) {//推进上层时间轮时间this.getOverflowWheel().advanceClock(timestamp);}}}
}

TimerTask

任务对象

/*** 任务*/
public class TimerTask {/*** 延迟时间*/private long delayMs;/*** 任务*/private Runnable task;/*** 时间槽*/protected TimerTaskList timerTaskList;/*** 下一个节点*/protected TimerTask next;/*** 上一个节点*/protected TimerTask pre;/*** 描述*/public String desc;public TimerTask(long delayMs, Runnable task) {this.delayMs = System.currentTimeMillis() + delayMs;this.task = task;this.timerTaskList = null;this.next = null;this.pre = null;}public Runnable getTask() {return task;}public long getDelayMs() {return delayMs;}@Overridepublic String toString() {return desc;}
}

TimerTaskList

任务集合

/*** 时间槽*/
public class TimerTaskList implements Delayed {/*** 过期时间*/private AtomicLong expiration = new AtomicLong(-1L);/*** 根节点*/private TimerTask root = new TimerTask(-1L, null);{root.pre = root;root.next = root;}/*** 设置过期时间*/public boolean setExpiration(long expire) {return expiration.getAndSet(expire) != expire;}/*** 获取过期时间*/public long getExpiration() {return expiration.get();}/*** 新增任务*/public void addTask(TimerTask timerTask) {synchronized (this) {if (timerTask.timerTaskList == null) {timerTask.timerTaskList = this;TimerTask tail = root.pre;timerTask.next = root;timerTask.pre = tail;tail.next = timerTask;root.pre = timerTask;}}}/*** 移除任务*/public void removeTask(TimerTask timerTask) {synchronized (this) {if (timerTask.timerTaskList.equals(this)) {timerTask.next.pre = timerTask.pre;timerTask.pre.next = timerTask.next;timerTask.timerTaskList = null;timerTask.next = null;timerTask.pre = null;}}}/*** 重新分配*/public synchronized void flush(Consumer<TimerTask> flush) {TimerTask timerTask = root.next;while (!timerTask.equals(root)) {this.removeTask(timerTask);flush.accept(timerTask);timerTask = root.next;}expiration.set(-1L);}@Overridepublic long getDelay(TimeUnit unit) {return Math.max(0, unit.convert(expiration.get() - System.currentTimeMillis(), TimeUnit.MILLISECONDS));}@Overridepublic int compareTo(Delayed o) {if (o instanceof TimerTaskList) {return Long.compare(expiration.get(), ((TimerTaskList) o).expiration.get());}return 0;}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/641604.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【vue3】GSAP在vue中的使用

一、获取GSAP npm install gsap 二、开始GSAP 导入GSAP&#xff0c;如果需要导入gsap的插件可以参考这里。 import gasp from gsap; 这里用的是选项式&#xff0c;在methods属性中创建一个方法用来写gsap的动画。 gasp_animation(){let tl gasp.timeline({defaults:{ ease:&…

【zlm】针对单个设备的码率的设置

目录 代码修改 实验数据一 实验数据二 同时拉一路视频后 修改记录 使用方法 代码修改 要被子类引用 &#xff0c;所以放在protected 不能放private 下面的结论&#xff0c;可以在下面的实验数据里引用。“同时拉一路视频后” 实验数据一 https://10.60.3.45:10443/index…

【学习】双线性插值

双线性插值公式 对于一个目的像素&#xff0c;设置坐标通过反向变换得到的浮点坐标为(iu,jv) (其中i、j均为浮点坐标的整数部分&#xff0c;u、v为浮点坐标的小数部分&#xff0c;是取值[0,1)区间的浮点数)&#xff0c;则这个像素得值 f(iu,jv) 可由原图像中坐标为 (i,j)、(i1…

php学习

php基础语法 一 php程序 1.php标记 开始标记<?php 和结束标记 ?>中间写 PHP 代码 当解析一个文件时&#xff0c;PHP 会寻找起始和结束标记&#xff0c;也就是告诉php 开始和停止解析二者之间的代码。此种解析方式使得PHP 可以被嵌入到各种不同的文档中去&#xff…

使用PSIM软件生成DSP28335流水灯程序

最近在学习DSP28335芯片&#xff0c;然后在使用PSIM仿真软件时发现这个仿真软件也支持28335芯片&#xff0c;于是就想学习下如何在PSIM软件中使用DSP28335芯片。在PSIM自带的官方示例中有使用DSP28335芯片的相关例子。 工程下载链接 https://download.csdn.net/download/qq_20…

2401llvm,clang插件

Clang插件 在编译时,Clang插件可运行额外的用户定义操作. 介绍 Clang插件在代码上运行FrontendActions.见FrontendAction教程,了解如何使用RecursiveASTVisitor编写FrontendAction. 这里,演示如何编写简单的clang插件. 编写PluginASTAction插件 与编写普通FrontendActions…

图解CART分类树评估器的参数

图解CART分类树评估器的参数

联邦学习:密码学 + 机器学习 + 分布式 实现隐私计算,破解医学界数据孤岛的长期难题

联邦学习&#xff1a;密码学 机器学习 分布式 提出背景&#xff1a;数据不出本地&#xff0c;又能合力干大事联邦学习的问题 分布式机器学习&#xff1a;解决大数据量处理的问题横向联邦学习&#xff1a;解决跨多个数据源学习的问题纵向联邦学习&#xff1a;解决数据分散在多…

(一)SpringBoot3---尚硅谷总结

目录 示例Demo&#xff1a; 1、我们先来创建一个空工程&#xff1a; 2、我们通过Maven来创建一个Module&#xff1a; 3、让此Maven项目继承父项目: 4、导入web开发的场景启动器 5、创建Springboot项目的主入口程序&#xff1a; 6、举例测试&#xff1a; 7、Springboot还能…

020-信息打点-红蓝队自动化项目资产侦察企查产权武器库部署网络空间

020-信息打点-红蓝队自动化项目&资产侦察&企查产权&武器库部署&网络空间 #知识点&#xff1a; 1、工具项目-红蓝队&自动化部署 2、工具项目-自动化侦查收集提取 3、工具项目-综合&网络空间&信息 演示案例&#xff1a; ➢自动化-武器库部署-F8x ➢自…

初级通信工程师-法律法规

1、《电信条例》修订的目的 ● 《电信条例》在最初颁布之后&#xff0c;陆续进行了两次修订&#xff0c;尤其是2016年做了最新的 修订。 ● 《电信条例》修订的目的是为了适应新的经济发展的需要&#xff0c;包括四个方面&#xff1a; ①规范电信市场秩序&#xff1b; ②维…

leetcode—图 岛屿数量

岛屿数量 给你一个由 1&#xff08;陆地&#xff09;和 0&#xff08;水&#xff09;组成的的二维网格&#xff0c;请你计算网格中岛屿的数量。 岛屿总是被水包围&#xff0c;并且每座岛屿只能由水平方向和/或竖直方向上相邻的陆地连接形成。 此外&#xff0c;你可以假设该网…

RabbitMQ入门篇【图文并茂,超级详细】

&#x1f973;&#x1f973;Welcome 的Huihuis Code World ! !&#x1f973;&#x1f973; 接下来看看由辉辉所写的关于Docker的相关操作吧 目录 &#x1f973;&#x1f973;Welcome 的Huihuis Code World ! !&#x1f973;&#x1f973; 前言 1.什么是MQ 2.理解MQ 3.生活…

记一个奇怪的万能密码

前言 打的站点打多了&#xff0c;什么奇怪的问题都会发生 打点 开局一个登录框 用户枚举到账号爆破 测了一下&#xff0c;没发现admin的弱口令&#xff0c;但是发现存在用户枚举漏洞&#xff0c;因此准备跑一下账号 输入密码为123456 进行账号爆破 成功爆破出账号 是的…

【ARM 嵌入式 编译系列 7.3 -- GCC 链接脚本中 DISCARD 与 .ARM.exidx】

请阅读【嵌入式开发学习必备专栏 之 ARM GCC 编译专栏】 文章目录 背景.ARM.exidx方法一:使用链接器脚本方法二:使用链接器选项注意事项背景 在移植 RT-Thread 到 cortex-m33(RA4M2)上的时候,在编译的时候遇到下面问题: Building target: ra4m2.elf arm

leetcode常见面试题总结 Python

持续维护中... 1. 寻找 k 个最小数 输入整数数组 arr &#xff0c;找出其中最小的 k 个数。例如&#xff0c;输入4、5、1、6、2、7、3、8这8个数字&#xff0c;则最小的4个数字是1、2、3、4。 示例&#xff1a; 输入&#xff1a;arr [3,2,1], k 2 输出&#xff1a;[1,2] …

Spring/Spring boot项目接入traceId

简介 分布式系统中&#xff0c;由多个服务构成&#xff0c;每个请求路由过来后&#xff0c;会在多个服务中留下追踪ID&#xff0c;可以基于此追踪ID排查问题&#xff0c;分析请求的执行链路。 业界也有比较成熟的链路追踪ID方案&#xff0c;比如Skywalking&#xff0c;它基于…

python 抓包tcp数据拷贝转发

在Python中&#xff0c;你可以使用scapy库进行抓包&#xff0c;使用shutil或io库进行数据的拷贝&#xff0c;以及使用socket库进行数据转发。下面是一个简单的示例&#xff0c;展示了如何进行这些操作&#xff1a; 首先&#xff0c;你需要安装必要的库。你可以使用pip来安装它…

Java算法 leetcode简单刷题记录4

Java算法 leetcode简单刷题记录4 买卖股票的最佳时机&#xff1a; https://leetcode.cn/problems/best-time-to-buy-and-sell-stock/ 笨办法&#xff1a; 记录当天的值及之后的最大值&#xff0c;相减得到利润&#xff1b; 所有的天都计算下&#xff0c;比较得到利润最大值&…

PyTorch视觉工具箱:图像变换与上采样技术详解(1)

目录 Pytorch中Vision functions详解 pixel_shuffle 用途 用法 使用技巧 注意事项 参数 数学理论公式 示例代码及输出 pixel_unshuffle 用途 用法 使用技巧 注意事项 参数 数学理论公式 示例代码及输出 pad 用途 用法 使用技巧 注意事项 参数 示例代码…