多线程实现方式---实现Runnable接口

一个类如果需要具备多线程的能力,也可以通过实现java.lang.Runnable接口进行实现。按照Java语言的语法,一个类可以实现任意多个接口,所以该种实现方式在实际实现时的通用性要比前面介绍的方式好一些。

使用实现Runnable接口实现多线程的示例代码如下:

/**

* 测试类

*/

public class Test2 {

public static void main(String[] args) {

//创建对象

MyRunnable mr = new MyRunnable();

Thread t = new Thread(mr);

//启动

t.start();

try{

for(int i = 0;i < 10;i++){

Thread.sleep(1000);

System.out.println("main:" + i);

}

}catch(Exception e){}

}

}

/**

* 使用实现Runnable接口的方式实现多线程

*/

public class MyRunnable implements Runnable {

public void run() {

try{

for(int i = 0;i < 10;i++){

Thread.sleep(1000);

System.out.println("run:" + i);

}

}catch(Exception e){}

}

}

该示例代码实现的功能和前面实现的功能相同。在使用该方式实现时,使需要实现多线程的类实现Runnable,实现该接口需要覆盖run方法,然后将需要以多线程方式执行的代码书写在run方法内部或在run方法内部进行调用。

在需要启动线程的地方,首先创建MyRunnable类型的对象,然后再以该对象为基础创建Thread类的对象,最后调用Thread对象的start方法即可启动线程。代码如下:

//创建对象

MyRunnable mr = new MyRunnable();

Thread t = new Thread(mr);

//启动

t.start();

在这种实现方式中,大部分和前面介绍的方式类似,启动的代码稍微麻烦一些。这种方式也是实现线程的一种主要方式。

12.2.3使用Timer和TimerTask组合

最后一种实现多线程的方式,就是使用java.util包中的Timer和TimerTask类实现多线程,使用这种方式也可以比较方便的实现线程。

在这种实现方式中,Timer类实现的是类似闹钟的功能,也就是定时或者每隔一定时间触发一次线程。其实,Timer类本身实现的就是一个线程,只是这个线程是用来实现调用其它线程的。而TimerTask类是一个抽象类,该类实现了Runnable接口,所以按照前面的介绍,该类具备多线程的能力。

在这种实现方式中,通过继承TimerTask使该类获得多线程的能力,将需要多线程执行的代码书写在run方法内部,然后通过Timer类启动线程的执行。

在实际使用时,一个Timer可以启动任意多个TimerTask实现的线程,但是多个线程之间会存在阻塞。所以如果多个线程之间如果需要完全独立运行的话,最好还是一个Timer启动一个TimerTask实现。

使用该种实现方式实现的多线程示例代码如下:

import java.util.*;

/**

* 测试类

*/

public class Test3 {

public static void main(String[] args) {

//创建Timer

Timer t = new Timer();

//创建TimerTask

MyTimerTask mtt1 = new MyTimerTask("线程1:");

//启动线程

t.schedule(mtt1, 0);

}

}

import java.util.TimerTask;

/**

* 以继承TimerTask类的方式实现多线程

*/

public class MyTimerTask extends TimerTask {

String s;

public MyTimerTask(String s){

this.s = s;

}

public void run() {

try{

for(int i = 0;i < 10;i++){

Thread.sleep(1000);

System.out.println(s + i);

}

}catch(Exception e){}

}

}

在该示例中,MyTimerTask类实现了多线程,以多线程方式执行的代码书写在该类的run方法内部,该类的功能和前面的多线程的代码实现类似。

而在该代码中,启动线程时需要首先创建一个Timer类的对象,以及一个MyTimerTask线程类的兑现,然后使用Timer对象的schedule方法实现,启动线程的代码为:

//创建Timer

Timer t = new Timer();

//创建TimerTask

MyTimerTask mtt1 = new MyTimerTask("线程1:");

//启动线程

t.schedule(mtt1, 0);

其中schedule方法中的第一个参数mtt1代表需要启动的线程对象,而第二个参数0则代表延迟0毫秒启动该线程,也就是立刻启动。

由于schedule方法比较重要,下面详细介绍一下Timer类中的四个schedule方法:

1、 public void schedule(TimerTask task,Date time)

该方法的作用是在到达time指定的时间或已经超过该时间时执行线程task。例如假设t是Timer对象,task是需要启动的TimerTask线程对象,后续示例也采用这种约定实现,则启动线程的示例代码如下:

Date d = new Date(2009-1900,10-1,1,10,0,0);

t. schedule(task,d);

则该示例代码的作用是在时间达到d指定的时间或超过该时间(例如2009年10月2号)时,启动线程task。

2、 public void schedule(TimerTask task, Date firstTime, long period)

该方法的作用是在时间到达firstTime开始,每隔period毫秒就启动一次task指定的线程。示例代码如下:

Date d = new Date(2009-1900,10-1,1,10,0,0);

t. schedule(task,d,20000);

该示例代码的作用是当时间达到或超过d指定的时间以后,每隔20000毫秒就启动一次线程task,这种方式会重复触发线程。

3、 public void schedule(TimerTask task,long delay)

该方法和第一个方法类似,作用是在执行schedule方法以后delay毫秒以后启动线程task。示例代码如下:

t. schedule(task,1000);

该示例代码的作用是在执行该行启动代码1000毫秒以后启动一次线程task。

4、 public void schedule(TimerTask task,long delay,long period)

该方法和第二个方法类似,作用是在执行schedule方法以后delay毫秒以后启动线程task,然后每隔period毫秒重复启动线程task。

例外需要说明的是Timer类中启动线程还包含两个scheduleAtFixedRate方法,这两个方法的参数和上面的第二个和第四个一致,其作用是实现重复启动线程时的精确延时。对于schedule方法来说,如果重复的时间间隔是1000毫秒,则实际的延迟时间是1000毫秒加上系统执行时消耗的时间,例如为5毫秒,则实际每轮的时间间隔为1005毫秒。而对于scheduleAtFixedRate方法来说,如果设置的重复时间间隔为1000毫秒,系统执行时消耗的时间为5毫秒,则延迟时间就会变成995毫秒,从而保证每轮间隔为1000毫秒。

介绍完了schedule方法以后,让我们再来看一下前面的示例代码,如果在测试类中启动两个MyTimerTask线程,一种实现的代码为:

import java.util.Timer;

/**

* 测试类

*/

public class Test4 {

public static void main(String[] args) {

//创建Timer

Timer t = new Timer();

//创建TimerTask

MyTimerTask mtt1 = new MyTimerTask("线程1:");

MyTimerTask mtt2 = new MyTimerTask("线程2:");

//启动线程

System.out.println("开始启动");

t.schedule(mtt1, 1000);

System.out.println("启动线程1");

t.schedule(mtt2, 1000);

System.out.println("启动线程2");

}

}

在该示例代码中,使用一个Timer对象t依次启动了两个MyTimerTask类型的对象mtt1和mtt2。而程序的执行结果是:

开始启动

启动线程1

启动线程2

线程1:0

线程1:1

线程1:2

线程1:3

线程1:4

线程1:5

线程1:6

线程1:7

线程1:8

线程1:9

线程2:0

线程2:1

线程2:2

线程2:3

线程2:4

线程2:5

线程2:6

线程2:7

线程2:8

线程2:9

从程序的执行结果可以看出,在Test4类中mtt1和mtt2都被启动,按照前面的schedule方法介绍,这两个线程均会在线程启动以后1000毫秒后获得执行。但是从实际执行效果却可以看出这两个线程不是同时执行的,而是依次执行,这主要是因为一个Timer启动的多个TimerTask之间会存在影响,当上一个线程未执行完成时,会阻塞后续线程的执行,所以当线程1执行完成以后线程2才获得了执行。

如果需要线程1和线程2获得同时执行,则只需要分别使用两个Timer启动TimerTask线程即可,启动的示例代码如下:

import java.util.Timer;

/**

* 测试类

*/

public class Test5 {

public static void main(String[] args) {

//创建Timer

Timer t1 = new Timer();

Timer t2 = new Timer();

//创建TimerTask

MyTimerTask mtt1 = new MyTimerTask("线程1:");

MyTimerTask mtt2 = new MyTimerTask("线程2:");

//启动线程

System.out.println("开始启动");

t1.schedule(mtt1, 1000);

System.out.println("启动线程1");

t2.schedule(mtt2, 1000);

System.out.println("启动线程2");

}

}

在该示例中,分别使用两个Timer对象t1和t2,启动两个TimerTask线程对象mtt1和mtt2,两者之间不互相干扰,所以达到了同时执行的目的。

在使用上面的示例进行运行时,由于Timer自身的线程没有结束,所以在程序输出完成以后程序还没有结束,需要手动结束程序的执行。例如在Eclipse中可以点击控制台上面的红色“Teminate”按钮结束程序。

12.2.4 小结

关于线程的三种实现方式,就简单的介绍这么多。其实无论那种实现方式,都可以实现多线程,在语法允许的前提下,可以使用任何一种方式实现。比较而言,实现Runnable接口方式要通用一些。

只是从语法角度介绍线程的实现方式,还是无法体会到线程实现的奥妙,下面将通过几个简单的示例来体会线程功能的强大,并体会并发编程的神奇,从而能够进入并发编程的领域发挥技术的优势。

时间: 2024-11-09 10:17:59

多线程实现方式---实现Runnable接口的相关文章

java多线程编程中实现Runnable接口方法相对于继承Thread方法的优势

 java多线程创建方法http://blog.csdn.net/cjc211322/article/details/24999163  java创建多线程方法之间的区别http://blog.csdn.net/cjc211322/article/details/25000449 java多线程编程中实现Runnable接口方法相对于继承Thread方法的优势

创建线程的第二种方式------实现Runnable接口的方式

package cn.itcast.demo16.Demo07.Runnable; /** * @author newcityman * @date 2019/7/22 - 23:17 */public class RunnableImpl implements Runnable { @Override public void run() { for (int i = 0; i <20 ; i++) { System.out.println(Thread.currentThread().getN

探Java多线程Thread类和Runnable接口之间的联系

首先复习一下Java多线程实现机制,Java实现多线程方法有如下这么几种: 1.继承了(extends)Thread类 2.实现了(implements)Runnable接口 也就是说  有如下两种情况 情况1: 继承Thread类.重写其方法run() .    然后new之.调用Start()方法 1 public class TestThread 2 { 3 private int i; 4 public static void main(String[] args) 5 { 6 // T

多线程-----Thread类与Runnable接口的区别

第一个继承Thread类来实现多线程,其实是相当于拿出三件事即三个卖早餐10份的任务分别分给三个窗口,他们各做各的事各卖各的早餐各完成各的任务,因为MyThread继承Thread类,所以在newMyThread的时候在创建三个对象的同时创建了三个线程:实现Runnable的, 相当于是拿出一个卖早餐10份的任务给三个人去共同完成,newMyThread相当于创建一个任务,然后实例化三个Thread,创建三个线程即安排三个窗口去执行. 一个类只能继承一个父类,存在局限:一个类可以实现多个接口.在

Java 多线程之 Thread 类 和 Runnable 接口初步使用

Thread 类 Thread 类是在 JDK1.0 时就存在的, 在 Java 中接触的多线程开发基本上都会从这个类开始. Thread之定义线程类 使用 Thread 创建线程的方法很简单, Thread 是一个类, 在需要创建线程时, 我们只需要继承这个类, 并将 run() 方法进行重写即可. class PrimeThread extends Thread { long minPrime; PrimeThread(long minPrime) { this.minPrime = min

2018-08-25多线程Thread类+Runnable接口+线程的6种状态

多线程: 进程:进程指正在运行的程序:确切的来说,当一个程序进入内存运行,即变成一个进程,进程是处于运行过程中的程序,并且具有一定独立功能(进入内存运行的程序成为进程)! 线程:线程是进程中的一个执行单元,负责当前进程中程序的执行,一个进程中至少有一个线程!一个进程中是可以有多个线程的,这个应用程序也可以称之为多线程程序(线程是执行单元,一个进程可以包括多个线程,一个程序可以有多个进程)! 单线程程序:若有多个任务只能依次执行(这个任务执行完毕,下一个任务开始执行)!如:去网吧上网,网吧只能让一

实现Runnable接口创建多线程及其优势

实现Runnable接口创建多线程: 创建一个Runnable接口的实现类RunnableImpl: 主线程中: 其中,链式编程的Thread类的静态方法currentThread方法点getName是获取的是当前线程的名称: 运行结果: 线程抢占cpu资源是随机的,无法人为控制: 实现Runnable接口创建多线程的优势: 避免单继承,可以实现其他接口: 降低耦合性,增强程序扩展性: 如在创建一个Runnable接口的实现类RunnableImpl2: 即创建一个新的run方法创建了一个新的线

多线程创建方式及线程安全问题

1.创建线程方式 一:  创建线程方式一继承Thread类 public clsss MyThread extends Thread{ //重写run方法,设置线程任务 Run(){ } } main(){ new MyThread().start(); } 获取线程名称: Thread.currentThread()获取当前线程对象 Thread.currentThread().getName();获取当前线程对象的名称 二:创建线程方式-实现Runnable接口 创建线程的步骤. 1.定义类

Java之多线程创建方式

多线程的由来 我们在之前,学习的程序在没有跳转语句的前提下,都是由上至下依次执行,那现在想要设计一个程序,边打游戏边听歌,怎么设计?要解决上述问题,咱们得使用多进程或者多线程来解决. 多线程的好处: 提高程序运行效率,让CPU的 使用率更高. 多个线程之间互不影响 关于多线程的一些名词解释 并发:指两个或多个事件在同一个时间段内发生. 并行:指两个或多个事件在同一时刻发生(同时发生). 进程:是指一个内存中运行的应用程序,每个进程都有一个独立的内存空间,一个应用程序可以同时运行多个进程:进程也是