模拟实现Java中的计时器

news/2025/2/24 13:45:44

定时器是什么

定时器也是软件开发中的⼀个重要组件. 类似于⼀个 "闹钟". 达到⼀个设定的时间之后, 就执⾏某个指定好的代码. 前端/后端中都会用到计时器.

定时器是⼀种实际开发中⾮常常⽤的组件. ⽐如⽹络通信中, 如果对⽅ 500ms 内没有返回数据, 则断开连接尝试重连. ⽐如⼀个 Map, 希望⾥⾯的某个 key 在 3s 之后过期(⾃动删除). 类似于这样的场景就需要⽤到定时器.

标准库中的定时器

• 标准库中提供了⼀个 Timer 类. Timer 类的核⼼⽅法为 schedule .

• schedule 包含两个参数. 第⼀个参数指定即将要执⾏的任务代码, 第⼆个参数指定多⻓时间之后 执⾏ (单位为毫秒).

java">// 定时器的使用
public class Demo21 {
    public static void main(String[] args) {
        Timer timer = new Timer();
        // main 方法中调用 timer.schedule 方法时, 
        // 它只是将任务注册到 Timer 中,并告诉 Timer 
        // 在 3000 毫秒后执行这个任务。
        // 任务的执行是由 Timer 内部的守护线程完成的。
        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("hello 3");
            }
        }, 3000);

        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("hello 2");
            }
        }, 2000);

        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("hello 1");
            }
        }, 1000);
        System.out.println("程序开始执行!");
    }
}

模拟实现定时器 

那么该怎么解决呢?

 

java">class MyTimerTask {
    // 任务啥时候执行. 毫秒级的时间戳.
    private long time;
    // 任务具体是啥.
    private Runnable runnable;

    public long getTime() {
        return time;
    }

    public Runnable getRunnable() {
        return runnable;
    }

    public MyTimerTask(Runnable runnable, long delay) {
        // delay 是一个相对的时间差. 形如 3000 这样的数值.
        // 构造 time 要根据当前系统时间和 delay 进行构造.
        time = System.currentTimeMillis() + delay;
        this.runnable = runnable;

    }
}

// 定时器的本体
class MyTimer {
    // 使用优先级队列 来保存上述的N个任务
    private PriorityQueue<MyTimerTask> queue = new PriorityQueue<>();
    // 定时器的核心方法 就是把要执行的任务添加到队列中
    public void schedule(Runnable runnable, long delay) {
        MyTimerTask task = new MyTimerTask(runnable, delay);
        queue.offer(task);
    }
    // MyTimer 中还需要构造一个 "扫描线程", 一方面去负责监控队首元素是否到点了,
    // 是否应该执行;
    // 一方面当任务到点之后,就要调用这里的 Runnable 的 Run 方法来完成任务

    public MyTimer() {
        // 扫描线程
        Thread t1 = new Thread(() -> {
            // 不停地去扫描当前的队首元素
            while (true) {
                try {
                    if (queue.isEmpty()) {
                        continue;
                    }
                    MyTimerTask task = queue.peek();
                    long curTime = System.currentTimeMillis();
                    if (curTime > task.getTime()) {
                        // 假设当前时间是 14:01, 任务时间是 14:00, 
                        // 此时就意味着应该要执行这个任务了.
                        // 需要执行任务.
                        queue.poll();
                        task.getRunnable().run();
                    }else {
                        // 让当前线程休眠一下, 按照时间差来休眠.
                        Thread.sleep(task.getTime() - curTime);
                    }
                }catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });
        t1.start();
    }
}

上述代码写完了计时器的核心逻辑, 但是这份代码中还有几个关键性的问题. 

最后完整的模拟实现代码.

java">import java.util.PriorityQueue;
import java.util.Timer;

/**
 * Created with IntelliJ IDEA.
 * Description:
 * User: xiaotutu
 * Date: 2025-02-20
 * Time: 21:41
 */

class MyTimerTask implements Comparable<MyTimerTask>{
    // 任务啥时候执行. 毫秒级的时间戳.
    private long time;
    // 任务具体是啥.
    private Runnable runnable;

    public long getTime() {
        return time;
    }

    public Runnable getRunnable() {
        return runnable;
    }

    public MyTimerTask(Runnable runnable, long delay) {
        // delay 是一个相对的时间差. 形如 3000 这样的数值.
        // 构造 time 要根据当前系统时间和 delay 进行构造.
        time = System.currentTimeMillis() + delay;
        this.runnable = runnable;

    }

    @Override
    public int compareTo(MyTimerTask o) {
        // 认为时间小的, 优先级高. 最终时间最小的元素, 就会放到队首.
        // 怎么记忆, 这里是谁减去谁?? 不要记!! 记容易记错~~
        // 随便写一个顺序, 然后实验一下就行了.
        return (int) (this.time - o.time);
        // return (int) (o.time - this.time);
    }
}

// 定时器的本体
class MyTimer {
    // 使用优先级队列 来保存上述的N个任务
    private PriorityQueue<MyTimerTask> queue = new PriorityQueue<>();

    // 用来加锁的对象
    private Object locker = new Object();

    // 定时器的核心方法 就是把要执行的任务添加到队列中
    public void schedule(Runnable runnable, long delay) {
        synchronized (locker) {
            MyTimerTask task = new MyTimerTask(runnable, delay);
            queue.offer(task);
            // 每次来新的任务, 都唤醒一下之前的扫描线程. 
            // 好让扫描线程根据最新的任务情况, 重新规划等待时间.
            locker.notify();
        }
    }
    // MyTimer 中还需要构造一个 "扫描线程", 一方面去负责监控队首元素是否到点了, 
    // 是否应该执行;
    // 一方面当任务到点之后,就要调用这里的 Runnable 的 Run 方法来完成任务

    public MyTimer() {
        // 扫描线程
        Thread t1 = new Thread(() -> {
            // 不停地去扫描当前的队首元素
            while (true) {
                try {
                    synchronized (locker) {
                        while (queue.isEmpty()) {
                            // 注意, 当前如果队列为空, 此时就不应该去取这里的
                            // 元素. 此处使用 wait 等待更合适. 
                            // 如果使用 continue, 就会使这个线程
                            // while 循环运行的飞快,
                            // 也会陷入一个高频占用 cpu 的状态(忙等).
                            //continue;
                            locker.wait();
                        }
                        MyTimerTask task = queue.peek();
                        long curTime = System.currentTimeMillis();
                        if (curTime > task.getTime()) {
                            // 假设当前时间是 14:01, 任务时间是 14:00, 此时就
                            // 意味着应该要执行这个任务了.
                            // 需要执行任务.
                            queue.poll();
                            task.getRunnable().run();
                        }else {
                            // 让当前线程休眠一下, 按照时间差来休眠.
                            // Thread.sleep(task.getTime() - curTime);
                            locker.wait(task.getTime() - curTime);
                        }
                    }
                }catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });
        t1.start();
    }
}

public class Demo22 {
    public static void main(String[] args) {
        MyTimer timer = new MyTimer();
        timer.schedule(new Runnable() {
            @Override
            public void run() {
                System.out.println("hello 3");
            }
        }, 3000);
        timer.schedule(new Runnable() {
            @Override
            public void run() {
                System.out.println("hello 2");
            }
        }, 2000);
        timer.schedule(new Runnable() {
            @Override
            public void run() {
                System.out.println("hello 1");
            }
        }, 1000);
        System.out.println("程序开始运行");
    }
}


http://www.niftyadmin.cn/n/5864410.html

相关文章

flink operator v1.10部署flink v1.19.2

1 概述 flink集群能对接kubernetes、yarn等集群管系统&#xff0c;本文介绍flink对接kubernetes。 flink kubernetes operator&#xff08;网址&#xff1a;https://nightlies.apache.org/flink/flink-kubernetes-operator-docs-release-1.10/docs/concepts/overview/&#xf…

Linux CentOS 上 Ollama 的安装与部署:从入门到实践

Linux CentOS 上 Ollama 的安装与部署:从入门到实践 随着人工智能技术的快速发展,大语言模型(LLMs)的应用场景越来越广泛。Ollama 作为一个轻量级的 AI 模型管理工具,为开发者提供了便捷的模型部署和管理解决方案。本文将详细介绍如何在 Linux CentOS 系统上安装和部署 O…

补充:文件上传、下载传输给前端之直接传递图片二进制数据:网络中的图片、音频、视频等非字符数据的传输

文章目录 1 Base64编码传递图片、视频二进制数据2 后端直接传递图片二进制数据案例2.1 后端:创建专用DTO(推荐方案)2.2 前端处理建议1 Base64编码传递图片、视频二进制数据 前面我们在学习Java的IO流的时候讲过字节流和字符流。 字符的传输:前后端字符的传输JSON中就是直接…

vscode settings(一):全局| 用户设置常用的设置项

参考资料 Visual Studio Code权威指南 by 韩骏 一. 全局设置与用户设置 1.1 Vscode支持两种不同范围的设置 用户设置(User Settings)&#xff1a;这是一个全局范围的设置&#xff0c;会应用到所有的Visual Studio Code实例中。工作区设置(Workspace Settings)&#xff1a;设…

【多线程-第三天-NSOperation的练习-tableView异步下载网络图片-下载操作缓存池 Objective-C语言】

一、下载操作缓存池 1.下面我们来看操作缓存池,我们先演示一下问题,看看为什么要加这么一个操作缓存池,什么是操作缓存池,不用管呢,我们先来看啊,首先有什么问题, 看这个问题之前,我这儿写一个touch,点击屏幕的时候调用, 额,不能点击屏幕啊,因为现在屏幕点不着,我…

如何自由切换 Node.js 版本?

作为开发者&#xff0c;我们经常在不同的项目中需要使用不同版本的 Node.js。有时旧项目需要旧版本&#xff0c;而新项目则可能依赖最新的 Node.js 版本。手动安装、卸载、切换版本不仅麻烦&#xff0c;而且容易出错。为了解决这个问题&#xff0c;Node.js 社区提供了多种方便的…

【论文解读】《Training Large Language Models to Reason in a Continuous Latent Space》

论文链接 1. 背景与动机 语言空间与推理的矛盾 目前大多数大语言模型&#xff08;LLMs&#xff09;在解决复杂问题时采用链式思维&#xff08;Chain-of-Thought, CoT&#xff09;方法&#xff0c;即利用自然语言逐步推导出答案。然而&#xff0c;论文指出&#xff1a; 自然语言…

力扣——搜索二维矩阵

题目链接&#xff1a; 链接 题目描述&#xff1a; 思路&#xff1a; 可以发现&#xff0c;如果把每一行拼起来&#xff0c;就是一个递增的数组&#xff0c;可以在这个递增的数组上使用二分法找到target如果拼起来的某个元素索引是i&#xff0c;那它在二维矩阵里面的索引是【…