实现多线程里面的定时器

本文介绍了如何使用Java创建一个线程安全的定时器,通过优先级队列存储任务,并利用扫描线程不断检查任务执行时间。任务类需要实现Comparable接口以进行时间比较,同时在定时器中加入锁机制确保线程安全。最后,在main函数中实例化定时器并安排多个任务进行演示。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

首先创建一个定时器类MyTimer和一个任务类Task(具体任务,和多久开始跑任务)

 任务类Task先实现这些基本字段和构造方法

 定时器里面定义一个优先级队列来放这些任务(本质最小堆,把时间最短的任务放在最上面,达到拿取任务为最近任务目的),写一个schedule方法来把这些任务put进队列

说明一下BlockingQueue是线程安全的并且实现了带有优先级队列的版本

具体实现内容就是传入任务和多久时间的参数,里面实例化一个任务,再将任务put进最小堆里.

但由于不知道队首元素是否为时间最小的任务,所以需要在定时器里创建一个扫描线程来不停地扫描这个优先级阻塞队列队首是否为最小值(因为不知道操作系统是否在这段时间又添加了新任务而且时间还更早)

 用lambda表达式创建扫描线程,一个while(true)循环take取出队首元素  再把当前时间与设定时间比较(可能会报错,原因后面解释)根据结果判断时间是否到了来

可以使用Java的线程池来实现在Spring定时器循环中启动多线程。具体步骤如下: 1. 定义一个线程池,可以使用`java.util.concurrent.Executors`类中的静态方法创建线程池。 2. 在Spring定时器的方法中调用线程池的`submit()`方法,提交需要执行的任务。任务可以是实现了`java.lang.Runnable`接口的类或者是实现了`java.util.concurrent.Callable`接口的类。 3. 线程池会自动管理线程的数量,并且会自动创建新的线程来执行新的任务。如果任务队列中有多个任务等待执行,线程池会按照先进先出的顺序依次执行任务。 4. 如果需要停止或重启定时器,可以调用线程池的`shutdown()`方法来停止所有线程,并等待所有任务执行完成。如果需要重启定时器,可以重新创建一个新的线程池,并调用`submit()`方法提交任务。 下面是一个示例代码: ```java import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; public class MyTask implements Runnable { @Override public void run() { // 处理任务的代码 } } public class MyTimer { private ExecutorService executor = Executors.newFixedThreadPool(10); public void start() { // Spring定时器的方法 while (true) { executor.submit(new MyTask()); Thread.sleep(1000); } } public void stop() { executor.shutdown(); executor.awaitTermination(10, TimeUnit.SECONDS); } } ``` 在上面的代码中,`MyTask`类是需要执行的任务,`MyTimer`类是Spring定时器的类。`executor`是线程池对象,使用`newFixedThreadPool()`方法创建一个固定大小的线程池。`start()`方法是定时器的方法,在循环中调用`submit()`方法提交任务。`stop()`方法是停止定时器的方法,调用`shutdown()`方法停止所有线程,并等待所有任务执行完成。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值