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

前言

使用JAVA实现一个简单版本的定时器


一、阻塞队列

1.wait和notify

使用 wait 和 notify 实现阻塞队列以下特点:

  • 入队列的时候如果发现队列满了,就会阻塞。直到有其他线程调用出队列操作让队列中有空位之后,才能继续入队列。
  • 出队列的时候如果发现队列空了,就会阻塞。直到有其他线程调用出队列操作让队列中有空位之后,才能继续入队列。

使用 wait 和 notify 解决定时器忙等问题:
当前时间 curTime 和 任务时间 task.time 对比,如果 curTime是8:00, task.time 是9:00,定时器将会忙等一小时。
解决办法:扫描线程内部,加上wait,如上述wait(一小时) 代码阻塞在 wait 处避免了频繁占用CPU。如果在等待过程中,新来了一个8:30执行的任务,此时 wait 也需要被唤醒。

阻塞队列是一个先进先出的队列

2.基本操作

普通队列的实现方法:

  1. 基于链表
  2. 基于数组(此处使用该方法)

队列的基本操作:

    1.入队列
    2.出队列
    3.取队首元素

针对阻塞队列,只提供前两个操作,不支持取队首元素

代码如下(示例):

public class ThreadDemo8 {
    static class BlockingQueue {
        //创建 array数组实现队列
        private int[] array = new int[1000];
        private int head = 0;
        private int tail = 0;
        private int size = 0;

        //阻塞版本入队列
        public void put(int value) throws InterruptedException {
            //加锁
            synchronized (this) {
                //判断队列是否满了,满了就进行阻塞
                if(size == array.length) {
                    wait();  //加入异常
                }

                //把 value 放到队尾即可
                array[tail] = value;
                tail++;
                if (tail == array.length) {
                    tail = 0;
                }
                size++;

                notify();
            }
        }

        //阻塞版本出队列
        public int take() throws InterruptedException {
            int ret = -1;
            synchronized (this) {
                //判断队列是否为空,空了就进行阻塞
                if(size == 0) {
                    wait();
                }
                ret = array[head];
                head++;
                if(head == array.length) {
                    head = 0;
                }
                size--;

                notify();
            }
            return ret;
        }
    }

3.生产者消费者模型

示例:生产者消费者模型是一种典型的并发编程方式

操作步骤:

  • 创建两个线程,分别模拟生产者和消费者
  • 第一次,让消费者消费的快一些,给生产者线程增加sleep,让生产者生产慢一些
  • 预期结果:消费者线程会阻塞等待
public static void main(String[] args) {
        BlockingQueue blockingQueue = new BlockingQueue();
        //搞两个线程,分别模拟生产者和消费者
        //让消费者消费的快一些,生产者生产慢一些
        Thread producer = new Thread() {
            @Override
            public void run() {
                for(int i = 0 ; i < 10000; i++) {
                    try {
                        blockingQueue.put(i);
                        System.out.println("生产元素:"+ i);
                        //生产间隔500毫秒
                        Thread.sleep(500);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        };
        producer.start();

        Thread consumer = new Thread() {
            @Override
            public void run() {
                while (true) {
                    try {
                        int ret = blockingQueue.take();
                        System.out.println("消费元素:" + ret);
                        //暂时不要 sleep
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        };
        consumer.start();
    }
}

运行结果如图:

在这里插入图片描述

二、定时器

1.构成

  1. 使用一个Task类来描述一个要执行的任务。
    任务中包括两方面:具体执行什么?什么时间执行?
  2. 使用一个阻塞优先队列来组织这些Task任务
  3. 使用一个扫描线程,定时扫描队首元素。
  4. 有一个接口,让调用者安排任务给定时器。

2.实现过程

定时器四个部分
1、有一个类描述一个任务

    static class Task implements Comparable<Task>{
        //Runnable 中有一个 run 方法,就可以借助这个 run 方法来描述要执行的具体任务
        private Runnable command;
        // time 表示啥时候来执行 command是绝对时间(ms级别的时间戳)
        private long time;

2、有一个阻塞优先队列来组织这些任务

//带优先级的阻塞队列
private PriorityBlockingQueue<Task> queue = new PriorityBlockingQueue<>();  

3、有一个扫描线程,定时扫描队首元素

public Timer () {
            //创建线程  (定时扫描队首元素
            Worker worker = new Worker(queue);
            worker.start();
        }

该线程扩展在run方法中

public void run() {
            //实现具体的线程执行内容
            while(true) {
                try {
                    //1.先取出队首元素,检查时间是否到了
                    Task task = queue.take();
                    //获取当前时间
                    long curTime = System.currentTimeMillis();
                    //2.检查当前任务时间是否到了
                    if(task.time > curTime) {
                        //预期时间还没到,就把任务再塞回队列中
                        queue.put(task);
                    } else {  //时间到,直接执行 run
                        task.run();
                    }
                } catch (InterruptedException e) {
                    e.printStackTrace();
                    break;
                }

            }
        }

4、有一个接口,让调用者能安排任务给定时器

public void schedule(Runnable command, long after) {
            Task task = new Task(command,after);
            queue.put(task);
        }

3.完整代码

代码如下:

import java.util.PriorityQueue;
import java.util.concurrent.PriorityBlockingQueue;

public class ThreadDemo9 {
    //优先队列中的元素必须是可比较的
    //比较规则的指定主要是两种方式:
    //1.让 Task 实现 Comparable 接口
    //2.让优先队列构造的时候,传入一个比较器对象 (Comparator)
    static class Task implements Comparable<Task>{
        //Runnable 中有一个 run 方法,就可以借助这个 run 方法来描述要执行的具体任务
        private Runnable command;
        // time 表示啥时候来执行 command是绝对时间(ms级别的时间戳)
        private long time;

        //构造方法的参数表示:多少毫秒之后执行,(相对时间)
        //                这个after相对时间参数只是为了后续用起来方便
        public Task(Runnable command,long after) {
            this.command = command;
            this.time = System.currentTimeMillis() + after;
        }

        //执行任务的具体逻辑
        public void run() {
            command.run();
        }

        @Override
        public int compareTo(Task o) {
            //谁的时间小先执行
            return (int) (this.time - o.time);
        }
    }

    static class Worker extends Thread {
        private  PriorityBlockingQueue<Task> queue = null;
        private Object mailBox = null;

        //构造方法保证 run 里面可以取到 queue
        public Worker(PriorityBlockingQueue<Task> queue,Object mailBox) {
            this.queue = queue;
            this.mailBox = mailBox;
        }

        @Override
        public void run() {
            //实现具体的线程执行内容
            while(true) {
                try {
                    //1.先取出队首元素,检查时间是否到了
                    Task task = queue.take();
                    //获取当前时间
                    long curTime = System.currentTimeMillis();
                    //2.检查当前任务时间是否到了
                    if(task.time > curTime) {
                        //预期时间还没到,就把任务再塞回队列中
                        queue.put(task);
                        synchronized(mailBox) {
                        mailBox.wait(task.time - curTime);
                        }
                    } else {  //时间到,直接执行 run
                        task.run();
                    }
                } catch (InterruptedException e) {
                    e.printStackTrace();
                    break;
                }

            }
        }
    }

    static class Timer {
    //为了避免忙等,需要使用 wait 方法,使用单独对象辅助进行 wait
    private Object mailBox = new Object();
        //定时器的基本构成有三部分
        //1.用一个类来描述“任务”
        //2.用一个阻塞优先队列来组织若干个任务,让队首元素就是时间最早的任务
        //  如果队首元素时间未到,那么其他元素也肯定不能执行
        private PriorityBlockingQueue<Task> queue = new PriorityBlockingQueue<>();  //带优先级的阻塞队列
        //3.用一个线程来循环扫描当前的阻塞队列的队首元素,如果时间到就执行指定的任务
        public Timer () {
            //创建线程  (定时扫描队首元素
            Worker worker = new Worker(queue,mailBox);
            worker.start();
        }
        //4.还需要提供一个方法,让调用者能把任务给“安排” 进来
        //  schedule => 安排的意思
        public void schedule(Runnable command, long after) {
            Task task = new Task(command,after);
            queue.put(task);
            synchronized(mailBox) {
            mailBox.notify();
            }
        }
    }

    public static void main(String[] args) {
        Timer timer = new Timer();
        timer.schedule(new Runnable() {
            @Override
            public void run() {
                System.out.println("hehe");
                //每隔 2秒钟就输出一个hehe,使用定时器达到循环效果
                //timer.schedule(this,2000);
            }
        },2000);  //after:2000意思是程序等待 2 秒钟后执行
    }
}


总结

JAVA提供了大量能使我们快速便捷地处理程序的方法,本文在阻塞优先队列处理任务的过程使用了PriorityBlockingQueue自带的优先级阻塞队列。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值