多线程---定时器

这篇具有很好参考价值的文章主要介绍了多线程---定时器。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

什么是定时器?

定时器的功能和“闹钟”类似,代码中的定时器通常都是“多长时间之后,执行某个动作”。

定时器的实现

标准库中的定时器

        // 标准库的定时器  执行完任务之后线程不会退出  需要手动杀死线程
    public static void main(String[] args) {
        Timer timer = new Timer();
        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("执行第一次定时器任务");
            }
        },3000);

        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("执行第二次定时器任务");
            }
        },4000);

        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("执行第三次定时器任务");
            }
        },5000);

        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("执行第四次定时器任务");
            }
        },6000);

        System.out.println("定时器启动");
    }

注:

  1. 一个定时器可以安排多个任务。
  2. 调用schedule安排任务时,一定要重写run方法,明确任务内容
  3. 定时器开启后不会自动结束,得手动杀死进程

自己实现一个定时器

版本一:实现简单的定时器

  • schedule方法是用来描述任务内容的。第一个参数是:任务内容;第二个参数是:任务多长时间后执行
	public MyTask(Runnable runnable, long delay){
        this.runnable = runnable;
        this.time = System.currentTimeMillis() + delay;
    }

	public void schedule(Runnable runnable,long delay){
        MyTask myTask = new MyTask(runnable, delay); 
    }
  • 用什么数据结构来存储任务呢?

    我们首先要考虑的是要让任务设置时间最短的先执行,这就有个优先级的问题,所以使用优先级队列

    同时,我们通常都会使用多个线程来执行这些设置的任务,就要保证线程安全。

    综上,我们最后决定使用优先级阻塞队列来存储任务

  • 我们创建一个专门的线程来执行定时器里的任务文章来源地址https://www.toymoban.com/news/detail-715521.html

//最基础的定时器完成
class MyTimer{
    private PriorityBlockingQueue<MyTask> priorityBlockingQueue = new PriorityBlockingQueue<>();



    public MyTimer(){
        Thread thread = new Thread(() -> {
            while (true){
                try {
                    MyTask myTask = priorityBlockingQueue.take();

                    if (System.currentTimeMillis() >= myTask.getTime()){
                        myTask.getRunnable().run();
                    }else {
                        priorityBlockingQueue.put(myTask);

                    }

                } catch (InterruptedException e) {
                    e.printStackTrace();
                }


            }
        });
        thread.start();
    }

    private void schedule(Runnable runnable,long delay){
        MyTask myTask = new MyTask(runnable,delay);
        priorityBlockingQueue.put(myTask);
    }
}

版本二:解决thread线程的忙等问题

  • 当定时器启动后有一个问题:不管任务时间有多久,thread线程一直执行,会占用CPU资源。我们想个办法让它在任务时间还没到的时候停下来,就要使用wait-notify
class MyTimer{
    private PriorityBlockingQueue<MyTask> priorityBlockingQueue = new PriorityBlockingQueue<>();

    private Object locker = new Object();


    public MyTimer(){
        Thread thread = new Thread(() -> {
            while (true){
                try {
                    MyTask myTask = priorityBlockingQueue.take();

                    if (System.currentTimeMillis() >= myTask.getTime()){
                        myTask.getRunnable().run();
                    }else {
                        priorityBlockingQueue.put(myTask);
                        synchronized (locker){
                            locker.wait(myTask.getTime() - System.currentTimeMillis());
                        }

                    }

                } catch (InterruptedException e) {
                    e.printStackTrace();
                }


            }
        });
        thread.start();
    }

    private void schedule(Runnable runnable,long delay){
        MyTask myTask = new MyTask(runnable,delay);
        priorityBlockingQueue.put(myTask);
    }
}

版本三:解决漏掉任务的问题

  • 又一个问题:如果schedule线程中新添加了一个任务 执行时间比wait时间短 不就错过了嘛。使用notify
class MyTimer{
    private PriorityBlockingQueue<MyTask> priorityBlockingQueue = new PriorityBlockingQueue<>();

    private Object locker = new Object();


    public MyTimer(){
        Thread thread = new Thread(() -> {
            while (true){
                try {
                    MyTask myTask = priorityBlockingQueue.take();

                    if (System.currentTimeMillis() >= myTask.getTime()){
                        myTask.getRunnable().run();
                    }else {
                        priorityBlockingQueue.put(myTask);
                        synchronized (locker){
                            locker.wait(myTask.getTime() - System.currentTimeMillis());
                        }

                    }

                } catch (InterruptedException e) {
                    e.printStackTrace();
                }


            }
        });
        thread.start();
    }

    private void schedule(Runnable runnable,long delay){
        MyTask myTask = new MyTask(runnable,delay);
        priorityBlockingQueue.put(myTask);
        synchronized (locker){
            locker.notify();
        }
    }
}

版本四:解决notify先执行的问题

  • 如果notify先执行了 take在notify之前执行 也就是说放进去了新的元素 但是take这边的线程没感觉到不就还没起到效果嘛。 扩大锁的范围
class MyTimer{
    private PriorityBlockingQueue<MyTask> priorityBlockingQueue = new PriorityBlockingQueue<>();

    private Object locker = new Object();


    public MyTimer(){
        Thread thread = new Thread(() -> {
            while (true){
                try {
                    synchronized (locker) {
                        MyTask myTask = priorityBlockingQueue.take();

                        if (System.currentTimeMillis() >= myTask.getTime()) {
                            myTask.getRunnable().run();
                        } else {
                            priorityBlockingQueue.put(myTask);

                            locker.wait(myTask.getTime() - System.currentTimeMillis());


                        }
                    }

                } catch (InterruptedException e) {
                    e.printStackTrace();
                }


            }
        });
        thread.start();
    }

    public void schedule(Runnable runnable,long delay){

        MyTask myTask = new MyTask(runnable, delay);
        priorityBlockingQueue.put(myTask);
        //这里起到的作用就是 刷新一下  看看队列中有没有执行时间更早的任务   有两种情况
        //1. 先放进去再刷新查看    先放进去最开始没注意到也没事儿  刷新的时候就会注意到重新计算wait时间
        //2. 先查看再放进去        会死锁(把notify的锁范围放大)   在一开始队列里没有数据的时候  会等待添加元素 但添加元素得获取到锁  此时阻塞的时候已经锁上了take
        //   没有人释放锁 所以 死锁了
        synchronized (locker) {
            locker.notify();
        }
    }
}

到了这里,关于多线程---定时器的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Java多线程案例之定时器

    定时器是一种实际开发中非常常用的组件, 类似于一个 “闹钟”, 达到一个设定的时间之后, 就执行某个指定好的代码. 比如网络通信中, 如果对方 500ms 内没有返回数据, 则断开连接尝试重连. 比如一个 Map, 希望里面的某个 key 在 3s 之后过期(自动删除). 类似于这样的场景就需要

    2024年02月20日
    浏览(49)
  • 【Java | 多线程案例】定时器的实现

    个人主页:兜里有颗棉花糖 欢迎 点赞👍 收藏✨ 留言✉ 加关注💓本文由 兜里有颗棉花糖 原创 收录于专栏【Java系列专栏】【JaveEE学习专栏】 本专栏旨在分享学习JavaEE的一点学习心得,欢迎大家在评论区交流讨论💌 Java中, Timer类 是用于计划和执行重复任务的类( Java标准

    2024年02月03日
    浏览(43)
  • 【Java】多线程案例(单例模式,阻塞队列,定时器,线程池)

    ❤️ Author: 老九 ☕️ 个人博客:老九的CSDN博客 🙏 个人名言:不可控之事 乐观面对 😍 系列专栏: 单例模式是设计模式之一。代码当中的某个类,只能有一个实例,不能有多个。单例模式分为:饿汉模式和懒汉模式 饿汉模式表示很着急,就想吃完饭剩下很多碗,然后一

    2024年02月06日
    浏览(44)
  • Java 多线程系列Ⅳ(单例模式+阻塞式队列+定时器+线程池)

    设计模式就是软件开发中的“棋谱”,软件开发中也有很多常见的 “问题场景”。针对这些问题场景,大佬们总结出了一些固定的套路。按照这些套路来实现代码可能不会很好,但至少不会很差。当前阶段我们需要掌握两种设计模式: (1)单例模式 (2)工厂模式 概念/特征

    2024年02月09日
    浏览(57)
  • 【Java|多线程与高并发】定时器(Timer)详解

    在Java中,定时器 Timer 类是用于执行定时任务的工具类。它允许你安排一个任务在未来的某个时间点执行,或者以固定的时间间隔重复执行。 在服务器开发中,客户端向服务器发送请求,然后等待服务器响应. 但服务器什么时候返回响应,并不确定. 但也不能让客户端一直等下去

    2024年02月07日
    浏览(45)
  • JAVAEE-定时器案例

    设置一个时间,当时间到了的时候,定时器就会去自动执行某个逻辑. 这里timer里面的任务执行完了也并不会结束,因为他并不知道是否还会添加新的任务进去, 处在严阵以待的状态 此时如果我们想要结束,应该使用cancel主动去结束. 1)需要有一个线程,负责掐时间,等任务到达合适的

    2024年02月05日
    浏览(47)
  • 【多线程】定时器,详解定时器原理,让大家更深刻的理解多线程

    前言: 大家好,我是 良辰丫 ,今天我们一起了解一下定时器,通过定时器来熟悉一下线程安全等相关知识点.💞💞💞 🧑个人主页:良辰针不戳 📖所属专栏:javaEE初阶 🍎励志语句:生活也许会让我们遍体鳞伤,但最终这些伤口会成为我们一辈子的财富。 💦期待大家三连,关注

    2024年02月01日
    浏览(60)
  • 多线程---定时器

    定时器的功能和“闹钟”类似,代码中的定时器通常都是“多长时间之后,执行某个动作”。 注: 一个定时器可以安排多个任务。 调用schedule安排任务时,一定要重写run方法,明确任务内容 定时器开启后不会自动结束,得手动杀死进程 版本一:实现简单的定时器 schedule方法

    2024年02月08日
    浏览(39)
  • 多线程案例(3)-定时器

    大家好,我是晓星航。今天为大家带来的是 多线程案例三 相关的讲解!😀 定时器是什么 定时器也是软件开发中的一个重要组件. 类似于一个 “闹钟”. 达到一个设定的时间之后, 就执行某个指定 好的代码. 定时器是一种实际开发中非常常用的组件. 比如网络通信中, 如果对方

    2024年02月14日
    浏览(42)
  • 多线程案例 | 单例模式、阻塞队列、定时器、线程池

    单例模式 单例模式是 设计模式 的一种 什么是设计模式? 设计模式好比象棋中的 “棋谱”,红方当头炮,黑方马来跳,针对红方的一些走法,黑方应招的时候有一些固定的套路,按照套路来走局势就不会吃亏,也就发明了一组\\\"棋谱\\\",称为设计模式 软件开发中也有很多常见

    2024年02月15日
    浏览(48)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包