当前位置: 首页 > news >正文

手搓一个定时器

目录

1.什么是定时器

2.计时器的使用

3.手搓定时器

3.1定义一个TimerTask类

3.2定义一个Timer类

3.3实现schedule方法

3.4实现Timer的构造方法

3.4.1随时随地查看优先级队列中是否有任务要执行

3.4.2获取队首任务,并判断是否到执行时间

3.4.3到达执行时间执行任务

4.整体的一个代码:


                    ​​​​​​​        ​​​​​​​        ​​​​​​​        ​​​​​​​        

1.什么是定时器

在看警匪片时,经常会有匪徒在银行安装定时炸弹,而这里的定时炸弹就可以看成一个定时器,当设定的时间到时,炸弹就会爆炸,而我们这里的定时器是当时间到时,就会执行相应的任务

2.计时器的使用

在Java中,关于计时器的类时Timer

Timer类是java.util包提供的API。

Timer中最主要的方法是schedule方法,他是将我们的任务,和时间放进一个数组模拟的队列中,然后在指定时间执行指定的任务

代码示例:

import java.util.Timer;
import java.util.TimerTask;public class Main19 {public static void main(String[] args) {Timer timer=new Timer();timer.schedule(new TimerTask() {@Overridepublic void run() {System.out.println("hello 3000");}},3000);timer.schedule(new TimerTask() {@Overridepublic void run() {System.out.println("hello 2000");}},2000);timer.schedule(new TimerTask() {@Overridepublic void run() {System.out.println("hello 1000");}},1000);System.out.println("hello main");}
}

schedule的第一个参数类型是TimerTask,这个类他继承了Runnable类,然后第二个参数是,当前程序运行多少时间执行这个任务,单位是ms

3.手搓定时器

简单介绍完java中的Timer的使用,下面我们就来模拟实现一个定时器

整体的一个思路:

1.定义任务类(TimerTask):用来存储每一个任务的属性,如执行的时间,以及要执行的任务

2.定义一个Timer类(定时器):每一个任务执行的先后顺序,以及执行过程中的一些细节问题

3.1定义一个TimerTask类

class TimerTask{private Runnable task;private long time;public TimerTask(Runnable task, long time) {this.task = task;this.time = time;}
}

在这个类中成员属性分别为要执行的任务,以及任务执行的时间,还有一个构造方法

3.2定义一个Timer类

class Timer{private PriorityQueue<TimerTask>queue=new PriorityQueue<>();public void schedule(Runnable task,long time){}public Timer() {Thread t=new Thread(()->{});t.start(); }}

1.在这个类中有一个优先级队列,这个是用来给各个任务执行的先后顺序进行排序的

2.schedule是用来将我们要执行的任务添加到优先级队列当中的

3.Timer()这个构造方法,是我们在创建这个类时,他就开启了一个线程,这个线程是用来按时间执行我们队列中的任务的,如果线程中有任务他就在指定时间去执行,如果没有任务,他就阻塞等待

3.3实现schedule方法

锁对象:

  Object lock=new Object();

schedule方法: 

    public void schedule(Runnable task,long delay){synchronized (lock){TimerTask timerTask=new TimerTask(task, System.currentTimeMillis()+delay);queue.offer(timerTask);}}

1.我们实例化一个TimerTask的对象,然后将schedule的参数作为任务,以及时间传递给TimerTask的构造方法作为参数(注意的是:这里构造方法接收的时间是任务执行的时间,而schedule参数的时间是任务距离调用schedule调用时相差的时间),所以构造方法参数的时间,我们要用当前时间加上延迟的时间(获取当前时间:System.currentTimeMillis());

2.得到TimerTask这个对象后,我们要将这个对象添加到优先级队列当中

3.在实例化Timer时,构造方法也会启动一个线程,所以我们这是一个多线程的程序,为了保证线程安全,我们给其加上一个锁

因为我们向优先级队列中添加的是我们自定义的类,他不知道该怎么对齐进行排序,所以我们要对TimerTask这个类定义一个比较规则

 class TimerTask implements Comparable<TimerTask>{@Overridepublic int compareTo(TimerTask o) {return (int)(this.time-o.time);}
}

我们将TimerTask实现一个Comparable接口,并重写Comparaable接口里面的comparaeTo方法,又因为我们想让时间近的任务放在队首,所以我们的比较规则是为了实现一个小根堆,如果this.time-o.time是一个整数就会将小的任务放在前面,然而我们的时间是long类型,我们需要将this.time-o.time的值强转为int类型

3.4实现Timer的构造方法

3.4.1随时随地查看优先级队列中是否有任务要执行

    public Timer() {Thread t=new Thread(()->{while(true){try {synchronized (lock){while(queue.isEmpty()){lock.wait();}}} catch (InterruptedException e) {throw new RuntimeException(e);}}});t.start();}

1.第一个while循环是当一次任务执行完毕,再次查看队列中是否有要执行的任务

2.第二个while循环是查看队列中是否有任务,如果没有任务,我们就阻塞等待,直到schedule被调用时,我们这里的wait被唤醒,然后进行接下来的操作

3.wait一般搭配synchronized来使用保证线程安全

为了在队列为空时,向队列中添加任务后,需要将wait唤醒,所以在schedule方法中要加上notify

    public void schedule(Runnable task,long delay){synchronized (lock){TimerTask timerTask=new TimerTask(task, System.currentTimeMillis()+delay);queue.offer(timerTask);lock.notify();}}

3.4.2获取队首任务,并判断是否到执行时间

 TimerTask task=queue.peek();if(System.currentTimeMillis()<task.getTime()){lock.wait(task.getTime()-System.currentTimeMillis());}

1.获取队首任务,使用peek,而不使用poll

2.如果未到达执行时间,我们就阻塞等待任务执行时间和当前时间之间的时差

3.如果在这个阻塞等待的时间段内,使用schedule向队列中添加了一个比当前任务更早的任务,schedule中的notify会将我们这里的带时间的wait唤醒,然后重新获取队列中的队首任务

3.4.3到达执行时间执行任务

if(System.currentTimeMillis()<task.getTime()){lock.wait(task.getTime()-System.currentTimeMillis());}else {task.run();queue.poll();}

1.这里使用else,也就是当当前系统时间大于等于任务执行的时间,执行任务

2.如果不使用else,当在等待过程中,新插入一条任务,那么就会解除阻塞,执行run和poll,从而导致程序出错

task.run方法需要我们在TimerTask类中进行实现

    public void run(){task.run();}

4.整体的一个代码:

package Demo4;import java.util.PriorityQueue;class TimerTask implements Comparable<TimerTask>{private Runnable task;private long time;public TimerTask(Runnable task, long time) {this.task = task;this.time = time;}public void run(){task.run();}public long getTime() {return time;}@Overridepublic int compareTo(TimerTask o) {return (int)(this.time-o.time);}
}class Timer{private PriorityQueue<TimerTask>queue=new PriorityQueue<>();Object lock=new Object();public void schedule(Runnable task,long delay){synchronized (lock){TimerTask timerTask=new TimerTask(task, System.currentTimeMillis()+delay);queue.offer(timerTask);lock.notify();}}public Timer() {Thread t=new Thread(()->{while(true){try {synchronized (lock){while(queue.isEmpty()){lock.wait();}TimerTask task=queue.peek();if(System.currentTimeMillis()<task.getTime()){lock.wait(task.getTime()-System.currentTimeMillis());}else {
task.run();
queue.poll();}}} catch (InterruptedException e) {throw new RuntimeException(e);}}});t.start();}}public class Main10 {public static void main(String[] args) {Timer timer=new Timer();timer.schedule(new Runnable() {@Overridepublic void run() {System.out.println("hello 3000");}},3000);timer.schedule(new Runnable() {@Overridepublic void run() {System.out.println("hello 2000");}},2000);timer.schedule(new Runnable() {@Overridepublic void run() {System.out.println("hello 1000");}},1000);}
}

完结: 

        ​​​​​​​        ​​​​​​​        ​​​​​​​        ​​​​​​​        ​​​​​​​        ​​​​​​​        ​​​​​​​

相关文章:

手搓一个定时器

目录 1.什么是定时器 2.计时器的使用 3.手搓定时器 3.1定义一个TimerTask类 3.2定义一个Timer类 3.3实现schedule方法 3.4实现Timer的构造方法 3.4.1随时随地查看优先级队列中是否有任务要执行 3.4.2获取队首任务&#xff0c;并判断是否到执行时间 3.4.3到达执行时间…...

AI提示词工程优化Prompt-GPT使用手册(科普一键收藏史上最强攻略)

Prompt(提示)&#xff0c;最初是 NLP 研究者为下游任务设计出来的一种任务专属的输入形式或模板。在 ChatGPT 引发大语言模型新时代之后&#xff0c;Prompt 指与大模型交互输入的代称。 随着大模型的进展&#xff0c;Prompt Engineering是一个持久的探索过程。 目录 什么是提示…...

【数据结构】快速排序(三种实现方式)

目录 一、基本思想 二、动图演示&#xff08;hoare版&#xff09; 三、思路分析&#xff08;图文&#xff09; 四、代码实现&#xff08;hoare版&#xff09; 五、易错提醒 六、相遇场景分析 6.1 ❥ 相遇位置一定比key要小的原因 6.2 ❥ 右边为key&#xff0c;左边先走 …...

利用前向勾子获取神经网络中间层的输出并将其进行保存(示例详解)

代码示例&#xff1a; # 激活字典&#xff0c;用于保存每次的中间特征 activation {}# 将 forward_hook 函数定义在 upsample_v2 外部 def forward_hook(name):def hook(module, input, output):activation[name] output.detach()return hookdef upsample_v2(in_channels, o…...

CTF-RE 从0到N: S盒

S盒&#xff08;Substitution Box&#xff09; 是密码学中的一种替换表&#xff0c;用于对输入数据进行非线性变换&#xff0c;以增加加密过程的复杂性。它主要用于对称加密算法中&#xff08;例如AES、DES&#xff09;&#xff0c;作为加密轮次的一部分&#xff0c;对输入字节…...

MT-Pref数据集:包含18种语言的18k实例,涵盖多个领域。实验表明它能有效提升Tower模型在WMT23和FLORES基准测试中的翻译质量。

2024-10-10&#xff0c;由电信研究所、里斯本大学等联合创建MT-Pref数据集&#xff0c;它包含18种语言方向的18k实例&#xff0c;覆盖了2022年后的多个领域文本。通过在WMT23和FLORES基准测试上的实验&#xff0c;我们展示了使用MT-Pref数据集对Tower模型进行对齐可以显著提高翻…...

【C++ 真题】B2099 矩阵交换行

矩阵交换行 题目描述 给定一个 5 5 5 \times 5 55 的矩阵(数学上&#xff0c;一个 r c r \times c rc 的矩阵是一个由 r r r 行 c c c 列元素排列成的矩形阵列)&#xff0c;将第 n n n 行和第 m m m 行交换&#xff0c;输出交换后的结果。 输入格式 输入共 6 6 6 …...

AAPL: Adding Attributes to Prompt Learning for Vision-Language Models

文章汇总 当前的问题 1.元标记未能捕获分类的关键语义特征 如下图(a)所示&#xff0c; π \pi π在类聚类方面没有显示出很大的差异&#xff0c;这表明元标记 π \pi π未能捕获分类的关键语义特征。我们进行简单的数据增强后&#xff0c;如图(b)所示&#xff0c;效果也是如…...

MySQLDBA修炼之道-开发篇(一)

三、开发基础 1. 数据模型 1.1 关系数据模型介绍 关于NULL 如果某个字段的值是未知的或未定义的&#xff0c;数据库会提供一个特殊的值NULL来表示。NULL值很特殊&#xff0c;在关系数据库中应该小心处理。例如查询语句“select*from employee where 绩效得分<85 or>绩…...

Spring MVC 知识点全解析

Spring MVC 知识点全解析 Spring MVC 是一个基于 Java 的请求驱动的 Web 框架&#xff0c;属于 Spring 框架的一部分&#xff0c;广泛用于构建企业级 Web 应用程序。本文将详细阐述 Spring MVC 的核心知识点&#xff0c;包括其工作原理、关键组件、配置、请求处理、数据绑定、…...

python 基于FastAPI实现一个简易的在线用户统计 服务

简易在线用户统计服务 概述 这是一个基于Python的FastAPI框架实现的服务&#xff0c;用于统计客户端的心跳信息&#xff0c;并据此维护在线用户列表以及记录活跃用户数。 功能特性 心跳接收&#xff1a;接受来自客户端的心跳包&#xff0c;以更新客户端的状态。在线用户统计…...

glibc中xdr的一个bug

本人在64位linux服务器上(centos7)&#xff0c;发现xdr_u_long这个函数有个bug&#xff0c;就是数字的范围如果超过unsigned int的最大值(4294967295)时&#xff0c;xdr_u_long失败。 这个场景主要用在unix时间戳上面&#xff0c;比如一款软件&#xff0c;设置有效期为100年。…...

Android Framework定制sim卡插入解锁pin码的界面

文章目录 手机设置SIM卡pin码一、安卓手机二、苹果手机 Android Framework中SIM卡pin码代码定位pin码提示文本位置定位pin码java代码位置 定制pin码framework窗口数字按钮 手机设置SIM卡pin码 设置 SIM 卡 PIN 码可以提高手机的安全性&#xff0c;防止他人在未经授权的情况下使…...

cc2530 Basic RF 讲解 和点灯讲解(1_1)

1. Basic RF 概述 Basic RF 是 TI 提供的一套简化版的无线通信协议栈&#xff0c;旨在帮助开发者快速搭建无线通信系统。它基于 IEEE 802.15.4 标准的数据包收发&#xff0c;但只用于演示无线设备数据传输的基本方法&#xff0c;不包含完整功能的协议。Basic RF 的功能限制包括…...

Android H5页面性能分析策略

文章目录 引言一、拦截资源加载请求以优化性能二、通过JavaScript代码监控资源下载速度三、使用vConsole进行前端性能调试四、使用Chrome DevTools调试Android端五、通过抓包分析优化网络性能六、总结 引言 在移动应用开发中&#xff0c;H5页面的性能直接影响到用户体验。本文…...

【前端面试】Typescript

Typescript面试题目回答 Typescript有哪些常用类型? Typescript的常用类型包括&#xff1a; 基本类型&#xff1a;boolean&#xff08;布尔类型&#xff09;、number&#xff08;数字类型&#xff09;、string&#xff08;字符串类型&#xff09;。特殊类型&#xff1a;nul…...

程序语言的内存管理:垃圾回收GC(Java)、手动管理(C语言)与所有权机制(Rust)(手动内存管理、手动管理内存)

文章目录 程序语言的内存管理&#xff1a;垃圾回收、手动管理与所有权机制引言一、垃圾回收机制&#xff08;GC&#xff09;&#xff08;Java&#xff09;1. 什么是垃圾回收机制2. 垃圾回收的工作原理3. 优点与缺点4. 示例代码 二、手动管理内存的分配和释放&#xff08;C语言&…...

研究生论文学习记录

文献检索 检索论文的网站 知网&#xff1a;找论文&#xff0c;寻找创新点paperswithcode &#xff1a;这个网站可以直接找到源代码 直接再谷歌学术搜索 格式&#xff1a;”期刊名称“ 关键词 在谷歌学术搜索特定期刊的关键词相关论文&#xff0c;可以使用以下几种方法&#…...

毕业设计选题:基于Django+Vue的图书馆管理系统

开发语言&#xff1a;Python框架&#xff1a;djangoPython版本&#xff1a;python3.7.7数据库&#xff1a;mysql 5.7数据库工具&#xff1a;Navicat11开发软件&#xff1a;PyCharm 系统展示 系统首页 图书馆界面 图书信息界面 个人中心界面 后台登录界面 管理员功能界面 用户…...

#网络安全#NGSOC与传统SOC的区别

NGSOC是Next Generation Security Operation Center&#xff08;下一代安全运营中心&#xff09;的缩写。 NGSOC安全运营服务基于态势感知与安全运营平台来开展监测分析等一系列的服务工作&#xff0c;旨在通过专业、高效的运营服务工作&#xff0c;帮助用户尽可能发挥NGSOC作…...

云原生核心技术 (7/12): K8s 核心概念白话解读(上):Pod 和 Deployment 究竟是什么?

大家好&#xff0c;欢迎来到《云原生核心技术》系列的第七篇&#xff01; 在上一篇&#xff0c;我们成功地使用 Minikube 或 kind 在自己的电脑上搭建起了一个迷你但功能完备的 Kubernetes 集群。现在&#xff0c;我们就像一个拥有了一块崭新数字土地的农场主&#xff0c;是时…...

解锁数据库简洁之道:FastAPI与SQLModel实战指南

在构建现代Web应用程序时&#xff0c;与数据库的交互无疑是核心环节。虽然传统的数据库操作方式&#xff08;如直接编写SQL语句与psycopg2交互&#xff09;赋予了我们精细的控制权&#xff0c;但在面对日益复杂的业务逻辑和快速迭代的需求时&#xff0c;这种方式的开发效率和可…...

Cilium动手实验室: 精通之旅---20.Isovalent Enterprise for Cilium: Zero Trust Visibility

Cilium动手实验室: 精通之旅---20.Isovalent Enterprise for Cilium: Zero Trust Visibility 1. 实验室环境1.1 实验室环境1.2 小测试 2. The Endor System2.1 部署应用2.2 检查现有策略 3. Cilium 策略实体3.1 创建 allow-all 网络策略3.2 在 Hubble CLI 中验证网络策略源3.3 …...

HTML 列表、表格、表单

1 列表标签 作用&#xff1a;布局内容排列整齐的区域 列表分类&#xff1a;无序列表、有序列表、定义列表。 例如&#xff1a; 1.1 无序列表 标签&#xff1a;ul 嵌套 li&#xff0c;ul是无序列表&#xff0c;li是列表条目。 注意事项&#xff1a; ul 标签里面只能包裹 li…...

Typeerror: cannot read properties of undefined (reading ‘XXX‘)

最近需要在离线机器上运行软件&#xff0c;所以得把软件用docker打包起来&#xff0c;大部分功能都没问题&#xff0c;出了一个奇怪的事情。同样的代码&#xff0c;在本机上用vscode可以运行起来&#xff0c;但是打包之后在docker里出现了问题。使用的是dialog组件&#xff0c;…...

Android第十三次面试总结(四大 组件基础)

Activity生命周期和四大启动模式详解 一、Activity 生命周期 Activity 的生命周期由一系列回调方法组成&#xff0c;用于管理其创建、可见性、焦点和销毁过程。以下是核心方法及其调用时机&#xff1a; ​onCreate()​​ ​调用时机​&#xff1a;Activity 首次创建时调用。​…...

Java编程之桥接模式

定义 桥接模式&#xff08;Bridge Pattern&#xff09;属于结构型设计模式&#xff0c;它的核心意图是将抽象部分与实现部分分离&#xff0c;使它们可以独立地变化。这种模式通过组合关系来替代继承关系&#xff0c;从而降低了抽象和实现这两个可变维度之间的耦合度。 用例子…...

Python 实现 Web 静态服务器(HTTP 协议)

目录 一、在本地启动 HTTP 服务器1. Windows 下安装 node.js1&#xff09;下载安装包2&#xff09;配置环境变量3&#xff09;安装镜像4&#xff09;node.js 的常用命令 2. 安装 http-server 服务3. 使用 http-server 开启服务1&#xff09;使用 http-server2&#xff09;详解 …...

在树莓派上添加音频输入设备的几种方法

在树莓派上添加音频输入设备可以通过以下步骤完成&#xff0c;具体方法取决于设备类型&#xff08;如USB麦克风、3.5mm接口麦克风或HDMI音频输入&#xff09;。以下是详细指南&#xff1a; 1. 连接音频输入设备 USB麦克风/声卡&#xff1a;直接插入树莓派的USB接口。3.5mm麦克…...

二维FDTD算法仿真

二维FDTD算法仿真&#xff0c;并带完全匹配层&#xff0c;输入波形为高斯波、平面波 FDTD_二维/FDTD.zip , 6075 FDTD_二维/FDTD_31.m , 1029 FDTD_二维/FDTD_32.m , 2806 FDTD_二维/FDTD_33.m , 3782 FDTD_二维/FDTD_34.m , 4182 FDTD_二维/FDTD_35.m , 4793...