前面我们已经对多线程的基础知识有了一定的了解,那么接下来我们将要对多线程进一步深入的学习;但在学习之前我们还是要对传统的技术进行一次回顾,本章我们回顾的则是:传统线程技术和传统的定时器实现.

一、传统线程技术

1.创建方式

1、继承thread类

Thread t = new Thread(){
@Override
public void run() {
}
};
t.start();

2、实现Runnable接口

Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
while (true) {
}
}
});
t1.start();

3、实现Callable接口

ExecutorService pool = Executors.newCachedThreadPool();
Future future = pool.submit(new Callable() {
public Object call() throws Exception {
return null;
}
});

2.比较

1、Thread VS Runnable

  • java不支持多继承,允许实现多个接口。Thread是类,Runnable是接口
  • Runnable适合于资源的共享(多个Thread使用相同的Runnable)
  • public class Thread extends Object implements Runnable. Thread是Runnable接口的子类

2、Runnable VS Callable

  • Callable的 call() 方法可以返回Future类型结果和抛出异常,而Runnable的run()方法没有这些功能
  • Callable通常利用ExecutorService的submit方法去启动call方法,Runnable还可以通过Thread的run方法启动run方法

二、传统定时器Timer

1、创建

到点执行,参数(TimerTask task, Date time),或者(TimerTask task, long delay)延迟多久后执行

new Timer().schedule(new TimerTask() {
@Override
public void run() {
}
}, new Date());

延迟多久执行,然后定时执行,参数(TimerTask task, long delay, long period)或者(TimerTask task, Date firstTime, long period)到点执行,然后定时执行

new Timer().schedule(new TimerTask() {
@Override
public void run() {
}, 1000, 5000})

还有类似的scheduleAtFixedRate(TimerTask task, long delay, long period)scheduleAtFixedRate(TimerTask task, Date firstTime,long period)

2、schedule和scheduleAtFixedRate区别

  • 2个参数的schedule:如果当前时间超过第一次执行时间,则立即执行,否则到定时时间执行
  • 3个参数的schedule:如果当前时间超过第一次执行时间,则立即执行,否则到定时时间执行。下一个任务执行一定是在上一个任务执行完之后执行。下一次任务执行的时间需要看上一个任务执行多久exceTime及周期时间period,如果exceTime>period则立即执行,否则等待period时间再执行。
  • scheduleAtFixedRate:每个任务执行的时间应该是定下了的。如果中间有作业处理时间太长导致后面的不能如期定时执行,则会立即执行后面的作业,直到追上了某一个任务的定时。如果当前时间超过第一次执行时间,则后面所有的作业都会立即执行,直到追上了某一个任务的定时。因为fixed-rate,可能导致同一时间重复执行,所以TimerTask中的执行体需要考虑同步(不是很懂)

schedule示例:

new Timer().schedule(new TimerTask(){
public void run() {
try {
System.out.println("execute task! "+ dateFormatter.format(this.scheduledExecutionTime()));
Random random = new Random();
int slpTime = random.nextInt(3)*1000 + 4000;
System.out.println("exec time:" + slpTime);
Thread.sleep(slpTime);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},startDate, 5 * 1000);

输出结果:

execute task!  2017-12-14 23:26:22 // 当前时间超过了设定了首次执行时间,立即执行
exec time:1000 // 第一次执行时间为1s小于周期时间2s
execute task! 2017-12-14 23:26:24 // 所以第二次在第一次执行时间2s之后执行
exec time:2000 // 第三次次执行时间为2s刚好等于周期时间2s
execute task! 2017-12-14 23:26:26 // 所以第三次在第二次执行时间2s之后执行
exec time:3000 // 第三次执行时间为3s大于周期时间2s
execute task! 2017-12-14 23:26:29 // 所以第四次在第三次执行时间3s之后执行
exec time:1000 // 之后就类似
execute task! 2017-12-14 23:26:31
exec time:2000
execute task! 2017-12-14 23:26:33
exec time:3000
execute task! 2017-12-14 23:26:36
exec time:1000
execute task! 2017-12-14 23:26:38

scheduleAtFixedRate示例:

System.out.println("start time: " + dateFormatter.format(new Date()));
new Timer().scheduleAtFixedRate(new TimerTask(){
int i = 0;
int slpTimes[] = new int[] {2000,4000,100,100,100};
public void run() {
try {
System.out.println("execute task:" + i + "! "+ dateFormatter.format(this.scheduledExecutionTime()) + " now time: " + dateFormatter.format(new Date())) ;
int slpTime = slpTimes[i++%slpTimes.length];
System.out.println("exec time:" + slpTime);
Thread.sleep(slpTime);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},startDate, 2 * 1000);

输出结果:

start time: 2017-12-15 00:00:09                                     //开始执行时间未到定时
execute task:0! 2017-12-15 00:01:00 now time: 2017-12-15 00:01:00 //定时开始执行
exec time:2000 //刚好执行2s
execute task:1! 2017-12-15 00:01:02 now time: 2017-12-15 00:01:02 //所以第二次在规定时间执行
exec time:4000 //第二次执行2s,导致第三次延迟执行(第三次应该在2017-12-15 00:01:04执行)
execute task:2! 2017-12-15 00:01:04 now time: 2017-12-15 00:01:06 //第三次在2017-12-15 00:01:06执行
exec time:100 //第三次执行100ms
execute task:3! 2017-12-15 00:01:06 now time: 2017-12-15 00:01:06 //因为之前导致了延迟,需要追赶,所以立即执行,以下类似
exec time:100
execute task:4! 2017-12-15 00:01:08 now time: 2017-12-15 00:01:08
exec time:100
execute task:5! 2017-12-15 00:01:10 now time: 2017-12-15 00:01:10
exec time:2000
execute task:6! 2017-12-15 00:01:12 now time: 2017-12-15 00:01:12

3、Timer的缺陷

Timer的替代品ScheduledExecutorService,这个不在本文进行介绍,后面会进行阐述ScheduledExecutorService.

参考资料:

《多线程视频》张孝祥

JAVA多线程提高一:传统线程技术&传统定时器Timer的更多相关文章

  1. Java基础_死锁、线程组、定时器Timer

    一.死锁问题: 死锁是这样一种情形:多个线程同时被阻塞,它们中的一个或者全部都在等待某个资源被释放.由于线程被无限期地阻塞,因此程序不可能正常终止. 比如,线程一需要第一把所,此时锁处于空闲状态,给了 ...

  2. JAVA多线程提高三:线程范围内共享变量&ThreadLocal

    今天我们学习的是如何在线程自己的范围内达到变量数据的共享,而各个线程之间又是互相独立开来,各自维护的,即我们说的ThreadLocal的作用. 一.概念 可以将每个线程用到的数据与对应的线程号存放到一 ...

  3. Java并发基础02. 传统线程技术中的定时器技术

    传统线程技术中有个定时器,定时器的类是Timer,我们使用定时器的目的就是给它安排任务,让它在指定的时间完成任务.所以先来看一下Timer类中的方法(主要看常用的TimerTask()方法): 前面两 ...

  4. Java多线程-同步:synchronized 和线程通信:生产者消费者模式

    大家伙周末愉快,小乐又来给大家献上技术大餐.上次是说到了Java多线程的创建和状态|乐字节,接下来,我们再来接着说Java多线程-同步:synchronized 和线程通信:生产者消费者模式. 一.同 ...

  5. Java多线程(二) —— 线程安全、线程同步、线程间通信(含面试题集)

    一.线程安全 多个线程在执行同一段代码的时候,每次的执行结果和单线程执行的结果都是一样的,不存在执行结果的二义性,就可以称作是线程安全的. 讲到线程安全问题,其实是指多线程环境下对共享资源的访问可能会 ...

  6. Java多线程(一) —— 线程的状态详解

    一.多线程概述  1. 进程 是一个正在执行的程序.是程序在计算机上的一次运行活动. 每一个进程执行都有一个执行顺序.该顺序是一个执行路径,或者叫一个控制单元. 系统以进程为基本单位进行系统资源的调度 ...

  7. Java多线程(五)线程的生命周期

    点我跳过黑哥的卑鄙广告行为,进入正文. Java多线程系列更新中~ 正式篇: Java多线程(一) 什么是线程 Java多线程(二)关于多线程的CPU密集型和IO密集型这件事 Java多线程(三)如何 ...

  8. Java多线程(一) —— 传统线程技术

    一.传统线程机制 1. 使用类Thread实现 new Thread(){ @Override public void run() { while(true){ try{ Thread.sleep(2 ...

  9. 【java并发】传统线程技术中创建线程的两种方式

    传统的线程技术中有两种创建线程的方式:一是继承Thread类,并重写run()方法:二是实现Runnable接口,覆盖接口中的run()方法,并把Runnable接口的实现扔给Thread.这两种方式 ...

随机推荐

  1. asp.net如何实现负载均衡方案讨论

    请注意,本文内容分多次修改,如需阅读,请阅读完整,因为早期的观点是不太合理的,后面由于水平进步,已经做了修改! 我的目标是我一个人搭建一个负载均衡网站.不接受这是网络部,或者运维,或者系统部的事情,所 ...

  2. Java中final修饰符深入研究

    一.开篇 本博客来自:http://www.cnblogs.com/yuananyun/ final修饰符是Java中比较简单常用的修饰符,同时也是一个被"误解"较多的修饰符.对很 ...

  3. JAVA程序测试时用到的与内存测试有关的东西

    1.JVM启动参数 垃圾回收器调用情况参数,使用如下参数可以看到程序何时启动GC进行垃圾回收,和垃圾回收的详细信息. java Test -XX:+PrintGCDetails -XX:+PrintG ...

  4. excel表中判断A列与B列内容是否相同,相同的话在C列按条件输出!

    判断两列数据是否相同,有以下几个函数判断(做笔记于此,方便以后查找): 1.=IF(AND(A4=B4),"相同","") 在C列输出相同字符 2.=IF(A1 ...

  5. Java NIO:IO与NIO的区别 -阿里面试题

    一.概念 NIO即New IO,这个库是在JDK1.4中才引入的.NIO和IO有相同的作用和目的,但实现方式不同,NIO主要用到的是块,所以NIO的效率要比IO高很多.在Java API中提供了两套N ...

  6. 【Python】Python处理csv文件

    Python处理csv文件 CSV(Comma-Separated Values)即逗号分隔值,可以用Excel打开查看.由于是纯文本,任何编辑器也都可打开.与Excel文件不同,CSV文件中: 值没 ...

  7. 小程序 setData() 方法

    setData() 参数格式 字段 类型 必填 描述 最低版本 data Object 是 这次要改变的数据   callback Function 否 回调函数 1.5.0 callback 是一个 ...

  8. 辣鸡蒟蒻Klaier的一些计划

    需要熟练的东西:cdq分治,堆,树链剖分,tarjan及其它一些图论算法,网络流,kmp,字符串哈希,线段树主席树,树状数组,斜率优化dp 需要学的东西:lct,后缀数组,AC自动机,平衡树 球队收益 ...

  9. 【Java】自动获取某表某列的最大ID数

    使用场景: 当需要往数据库插入数据时,表的主键需要接着已经有的数据后面进行自增.比如已经wq_customer表里,主键为TBL_ID,如果是空表,那么插入的数据TBL_ID设置为1,如果已经有n条数 ...

  10. [ZJOI2014]力 FFT

    题面 题解: \[F_j = \sum_{i < j}\frac{q_iq_j}{(i - j)^2} - \sum_{i > j}{\frac{q_iq_j}{(i - j)^2}}\] ...