Java中定时器Timer致命缺点(附学习方法)
简介
这篇文章我一直在纠结到底要不要写,不想写一来因为定时器用法比较简单,二来是面试中也不常问。后来还是决定写了主要是想把自己分析问题思路分享给大家,让大家在学习过程中能够参考,学习态度我相信大部分人没有问题,特别是正在看我博文的小伙伴那更不用说了!!给你们点个狂力赞。接下来就是学习方法了,我发现近期来咨询我问题的小伙伴学习姿势不对,所以我用Java中定时器Timer为案例整理下我的学习方法。万丈高楼平地起,所以我一贯的做法都是先用

案例1:定时器打印Hello World!
import java.text.ParseException;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;
/**
* @author :jiaolian
* @date :Created in 2021-01-05 20:42
* @description:Timer启动后内置线程不销毁
* @modified By:
* 公众号:叫练
*/
public class TimerThreadNoStopTest {
//TimerTask为抽象类,继承TimerTask类必须要实现里面抽象方法
private static class Task extends TimerTask {
@Override
public void run() {
System.out.println("hello world!");
}
}
public static void main(String[] args) throws ParseException {
Timer timer = new Timer();
Task task = new Task();
long currenTime = System.currentTimeMillis();
//提交Task线程;程序按传入日期运行
timer.schedule(task,new Date(currenTime));
}
}
如上面程序代码,Timer提交了一个task任务并传入了currenTime当前时间,控制台马上打印了"hello world!",如果schedule传入的第二个参数是new Date(currenTime+2000)表示延迟2m执行task任务,这里简单使用方法就不过多的描述了,但是大家在学习过程中遇到疑惑的问题一定要多尝试多写代码测试,这是理解代码必不可少的一部分,不要以为能看懂就不写了,

线程不死问题?
原因分析:如下图所示,主线程执行Timer timer = new Timer();会创建了一个新的子线程timer,timer线程通过死循环来取队列里面的任务task[1],队列其实就是一个数组实现TaskQueue,队列里面如果没有任务,那timer线程就会一直等待直到主线程调用schedule提交任务,主线程就会将task加入到TaskQueue队列数组并通知timer线程执行任务并删除队列的第一个任务,如果是主线程提交的是定时任务,就会将任务重新加入队列,任务执行完毕后,如果此时队列为空,timer线程就会继续等待任务提交到队列,一直会循环上面的过程。如果想退出timer线程,可以

案例2:单线程问题
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;
/**
* @author :jiaolian
* @date :Created in 2021-01-06 10:53
* @description:多任务执行测试,任务只能顺序执行;
* @modified By:
* 公众号:叫练
*/
public class MultTaskExecuteTest {
private static final SimpleDateFormat SIMPLE_DATE_FORMAT = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
private static class MyTask1 extends TimerTask {
@Override
public void run() {
System.out.println("task1 begin:"+SIMPLE_DATE_FORMAT.format(new Date()));
try {
Thread.sleep(10000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("task1 end:"+SIMPLE_DATE_FORMAT.format(new Date()));
}
}
private static class MyTask2 extends TimerTask {
@Override
public void run() {
System.out.println("task2 begin:"+SIMPLE_DATE_FORMAT.format(new Date()));
System.out.println("task2 end:"+SIMPLE_DATE_FORMAT.format(new Date()));
}
}
public static void main(String[] args){
Timer timer = new Timer();
MyTask1 myTask1 = new MyTask1();
MyTask2 myTask2 = new MyTask2();
long curTime = System.currentTimeMillis();
System.out.println("当前时间:"+SIMPLE_DATE_FORMAT.format(curTime));
timer.schedule(myTask1,new Date(curTime));
//myTask1执行时间过长,myTask2 被执行时间会被延迟;
timer.schedule(myTask2,new Date(curTime+1000));
}
}
如上面程序代码,timer线程提交了两个任务myTask1,myTask2,myTask1任务会立刻执行,myTask2计划延迟一秒执行,myTask1执行过程中会休息10秒钟,我们观察任务执行时间如下图所示,myTask2任务是等待myTask1任务执行完毕后再执行的,其实myTask2只是延迟一秒执行,结果却延迟了10秒,说明了timer单线程会串行化任务导致myTask2延迟执行,所以

定时器实际应用场景
在日常系统开发中,相信你遇到过类似需要重复执行的任务,比如
学习方法心得
大家可以看到我最近几篇文章分析多线程花了不少精力都在谈论可见性,原子性,母鸡下蛋生成消费问题等问题,因为这些特性是理解多线程的基础,在我看来基础又特别重要,所以怎么反复写我认为都不过分,在这之前有很多新手或者有2到3年工作经验的童鞋经常会问我关于
总结
我们用代码简述timer定时器提交任务,并说明了timer是单线程的适合轻量级的定时任务,这是它的缺陷。鉴于篇幅有限其中timer还有很多方法我们没有用代码贴出来,比如定时执行,延迟执行,timer取消方法,希望大家一一对着书本执行起来,给大家推荐本多线程入门学习书籍,《Java多线程编程核心技术》,书本以案例为主,也没有特别难理解的案例,非常适合新手学习。如果需要pfd版请联系我!喜欢的请点赞加关注哦。我是
