线程并发->07Timer

一、参考文章:

二、Timer问题:

  • 1、如何开启任务;
  • 2、如何停止正在运行的任务;
  • 3、线程间通信;
  • 4、Timer有什么缺陷;

三、demo:

private Timer mTimer;

public void timerStart() {
    TimerTask timerTask = new TimerTask() {
        @Override
        public void run() {
            LogUtils.log(getClass(), "timerTask");
        }
    };
    mTimer = new Timer();
    mTimer.schedule(timerTask, 0, 500);
}

public void timerCancel() {
    mTimer.cancel();
}

四、Timer任务开启:

4.1 Timer构造函数:
public class Timer {

    private final TaskQueue queue = new TaskQueue();
    
    private final TimerThread thread = new TimerThread(queue);
    /**
     * 1. Timer初始化时, 创建一个final类型的TaskQueue和TimerThread;  
     * 2. TimerThread持有TaskQueue的引用, 然后直接调用thread.start方法, 目前猜测可能
     *    用到了生产者-消费者模式, 通过TaskQueue.add唤醒当前TimerThread;
     */
    public Timer() {
        this("Timer-" + serialNumber());
    }

    public Timer(String name) {
        thread.setName(name);
        /**
         * 开启任务线程, start方法会触发其内部的run方法执行, 这里也就是说一旦初始化
         * Timer, 就会开启任务线程; 
         */
        thread.start();    模块<4.2>
    }
}

class TaskQueue {
    void rescheduleMin(long newTime) {
        queue[1].nextExecutionTime = newTime;
        fixDown(1);
    }
}
4.2 TimerThread.run:
class TimerThread extends Thread {
    public void run() {
        try {
            mainLoop();
        } finally {
            synchronized(queue) {
                newTasksMayBeScheduled = false;
                queue.clear();  
            }
        }
    }

    private void mainLoop() {
        /**
         * 采用while-true的方式, 后续分析要重点关注如何退出while-true;<TODO>
         */
        while (true) {
            TimerTask task;
            boolean taskFired;
            /**
             * 这里的synchronized是为了<//2--->>的queue.wait使用;
             */
            synchronized(queue) {
                /**
                 * 1. newTasksMayBeScheduled表示当前Timer是否可用, 默认true表示可用;
                 * 2. 如果queue为空, 且Timer可用, 则进入<//2--->>挂起当前线程, 并且释放锁;
                 */
//1--->
                while (queue.isEmpty() && newTasksMayBeScheduled)
//2--->             /**
//                   * 如果当前没有要执行的任务, 则线程在这里被挂起;
//                   */
                    queue.wait();
                /**
                 * 执行if内的break需要以下两种条件:
                 *  1. queue.isEmpty() == true;
                 *  2. newTasksMayBeSchedule == false;
                 */
//3--->
                if (queue.isEmpty())
                    break;
                long currentTime, executionTime;
                /**
                 * 执行到这里说明当前TaskQueue不为empty, 从TaskQueue中取出TimerTask;
                 */
                task = queue.getMin();
                synchronized(task.lock) {
                    // 如果当前任务已经被取消, 则从TaskQueue中移除当前TimerTask;
                    if (task.state == TimerTask.CANCELLED) {
                        queue.removeMin();
                        continue;  
                    }
                    currentTime = System.currentTimeMillis();
                    executionTime = task.nextExecutionTime;
                    /**
                     * 结合模块<五>可知, executionTime为初始化TimerTask时指定的执行时间:
                     * 1. 如果executionTime > currentTime即TimerTask需要延时处理, taskFired = false,
                     *    跳转至<//4--->>进入阻塞状态;        
                     * 2. 如果executionTime ≤ currentTime即TimerTask不需要延时处理, taskFired = true,
                     *    跳转至<//5--->>执行run方法;
                     */
                    if (taskFired = (executionTime<=currentTime)) {
                        /**
                         * period相当于定时器的作用, 每隔period时间执行一次run方法; 分两种情况:
                         * 1. period > 0, 每隔period时间通过rescheduleMin刷新任务下一次的执行时间点;
                         * 2. period = 0, task执行完之后, 从queue中移除, 即TimerTask.run只会执行一次; 
                         */
                        if (task.period == 0) { // Non-repeating, remove
                            queue.removeMin();
                            task.state = TimerTask.EXECUTED;
                        } else { // Repeating task, reschedule
                            queue.rescheduleMin(
                                task.period<0 ? currentTime   - task.period
                                              : executionTime + task.period);
                        }
                    }
                }
//4--->
                if (!taskFired)
                    /**
                     * 当前线程被挂起executionTime - currentTime时间之后被唤醒继续向下执行;
                     */
                    queue.wait(executionTime - currentTime);
            }
//5--->   
            if (taskFired)  
                /**
                 * 1. 初始化Timer时, 会初始化一个final类型的TimerThread, 一个Timer有且仅有一个
                 *    TimerThread, 所以多个任务在这里进行串行执行, 这里就会有一个弊端, 模块<7>
                 *    使用demo进行说明;
                 * 2. 当前run方法执行完成以后, 再次执行while(true)尝试用queue中取出task执行;
                 */
                task.run();
        }
    }
}

五、Timer任务执行:

public class Timer {
    public void schedule(TimerTask task, long delay, long period) {
        sched(task, System.currentTimeMillis()+delay, -period);
    }

    private void sched(TimerTask task, long time, long period) {
        if (Math.abs(period) > (Long.MAX_VALUE >> 1))
            period >>= 1;

        synchronized(queue) {
            /**
             * thread.newTasksMayBeScheduled默认为true, 如果thread.newTasksMayBeScheduled
             * 为false(即模块<6>调用Timer.cancel方法), 再次调用Timer.schedule方法, 会抛出此异常;
             */
            if (!thread.newTasksMayBeScheduled)
                throw new IllegalStateException("Timer already cancelled.");
            /**
             * 对TimerTask进行初始化操作, state默认为SCHEDULED;
             */
            synchronized(task.lock) {
                if (task.state != TimerTask.VIRGIN)
                    throw new IllegalStateException("Task already scheduled or cancelled");
                task.nextExecutionTime = time;
                task.period = period;
                task.state = TimerTask.SCHEDULED;
            }

            queue.add(task);
            if (queue.getMin() == task)
                queue.notify();
        }
    }

    class TaskQueue {

        void add(TimerTask task) {
            if (size + 1 == queue.length)
                queue = Arrays.copyOf(queue, 2*queue.length);
            queue[++size] = task;
            fixUp(size);
        }
    
        TimerTask getMin() {
            return queue[1];
        }
    }
}

六、任务取消:

public void cancel() {
    /**
     * 这里用的锁对象与模块<4.2>run和模块<五>是同一个锁对象, 所以取消当前正在执行的任务一定是在
     * 当前任务执行完成之后, 或者在schedule完成之后才会执行;
     */
    synchronized(queue) {
        /**
         * 1. 调用cancel之后, newTasksMayBeScheduled = false, 响应模块<五>的schedule方法;
         * 2. 调用queue.clear()响应模块<4.2>的<//3--->>处由于queue.isEmpty()且
         *    newTasksMayBeScheduled = false触发break的执行导致跳出当前while(true);
         * 3. queue.notify是为了唤醒模块<4.2>当queue为空时, 通过queue.wait方式被挂起的线程, 
         *    此时线程被唤醒之后, 再次执行while(true)从而退出while(true)循环;
         */
        thread.newTasksMayBeScheduled = false;
        queue.clear();
        queue.notify();  // In case queue was already empty.
    }
}

七、反应模块<4.2>_<//5--->>的一个弊端代码:

public class TimerTest {

    private static long start;

    public static void timerTest() {
        TimerTask task1 = new TimerTask() {
            @Override
            public void run() {
                LogUtils.log(getClass(), "task1 invoked, " + (System.currentTimeMillis() - start));
                try {
                    Thread.sleep(3000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }

            }
        };
        TimerTask task2 = new TimerTask() {
            @Override
            public void run() {
                LogUtils.log(getClass(), "task2 invoked, " + (System.currentTimeMillis() - start));
            }
        };
        Timer timer = new Timer();
        start = System.currentTimeMillis();
        timer.schedule(task1, 1000);
        timer.schedule(task2, 3000);
    }
}
  • 这段代码取自张鸿洋的文章;
    打印结果如下:
05-06 19:58:06.286 4582-4648/com.test V/AndroidTest: ->task1 invoked, 1000
05-06 19:58:09.286 4582-4648/com.test V/AndroidTest: ->task2 invoked, 4001

为何会是这样一种结果?
模块<4.2>的<//5--->>部分说明了Timer内部只有一个线程, 所以通过schedule方式添加的TimerTask是以串行的方式进行, 这就导致task2必须要在task1执行完成以后才能执行, 而task1执行耗时包括delay(1000) + sleep(3000) = time(4000)

因此如果使用Timer连续执行多个task, 后面每个task执行的时间很可能会不准确;

  • 弊端的原因是因为每次提交任务之后, 任务是以串行的方式在TimerThread内部执行,
    而如果后继Task的执行启动时间小于前继Task的执行(启动时间+执行时间), 则后继
    任务的执行将会被推迟;

八、Timer如何保证多任务串行的方式执行:

8.1 Timer.schedule与TimerThread.run:
class TimerThread extends Thread {

    private void mainLoop() {
        while (true) {
            synchronized(queue) {
                while (queue.isEmpty() && newTasksMayBeScheduled)
                    queue.wait();
            }
        }
        if (taskFired) 
            task.run();
    }
} 
public class Timer{

    public void schedule(TimerTask task, long delay, long period) {
        sched(task, System.currentTimeMillis()+delay, -period);
    }

    private void sched(TimerTask task, long time, long period) {
        synchronized(queue) {
            queue.add(task);
            if (queue.getMin() == task)
                queue.notify();
        }
    }
}
  • 1、上面只列了代码片段, queue为final类型, mainLoop(Thread_1)和schedule(Thread_2)可以认为是消费者--生产者模式;
  • 2、mainLoop中如果queue.isEmpty == true, 则释放锁并挂起当前线程, 然后sched所处线程尝试获取锁, 添加task然后notify唤醒线程Thread_1;
  • 3、如果queue.isEmpty == false, 则Thread_1持有锁, Thread_2 执行sched时被挂起, 直到 Thread_1 执行完task.run之后才会释放锁, 然后Thread_2尝试获取锁像queue中添加Task;
  • 4、所以Task能被成功添加到queue中一定是满足queue.isEmpty == true 或者 Thread_1 执行task.run结束;
  • 5、所以Task(N)的实际启动时间 == Max(Task(N)启动时间, Task(N-1)启动时间 + Task(N - 1)执行时间);
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,245评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,749评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,960评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,575评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,668评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,670评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,664评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,422评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,864评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,178评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,340评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,015评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,646评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,265评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,494评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,261评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,206评论 2 352

推荐阅读更多精彩内容

  • Timer 定时器相信都不会陌生,之所以拿它来做源码分析,是发现整个控制流程可以体现很多有意思的东西。 在业务开发...
    石先阅读 6,341评论 2 13
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,019评论 25 707
  • 先运行一段测试代码 代码中生成一个间隔5s,tolerance为0.5s的NSTimer,加入主线程的RunLoo...
    jinqiushi阅读 5,298评论 11 47
  • 有一次,我跟妈妈去菜场买小葱,我忽然有了一个有趣的想法--种葱。 我先准备了花盆、泥土和水,接着倒入一半泥土放进花...
    3b7d71778f07阅读 474评论 1 1
  • 今天真正体验到,直面面对一切,所有的问题似乎都变简单了不少,恐慌与不安也会相应的减少,困难也并非想象中的难以克服。...
    大丝瓜君阅读 277评论 0 0