Java多线程(二)——常用的实现多线程的两种方式
一、继承Thread类创建线程类
Java使用Thread类代表线程,所有的线程对象都必须是Thread类或其子类的实例。每个线程的作用是完成一定的任务,实际上就是执行一段程序流即一段顺序执行的代码。Java使用线程执行体来代表这段程序流。
Thread类的声明如下:
public class Thread implements Runnable {}
可以看到,Thread本身就实现了Runnable接口。
Java中通过继承Thread类来创建并启动多线程的步骤如下:
01. 定义Thread类的子类,并重写该类的run()方法,该run()方法的方法体就代表了线程需要完成的任务,因此把run()方法称为线程执行体。
02. 创建Thread子类的实例,即创建了线程对象。
03. 调用线程对象的start()方法来启动该线程。
示例:
package com.demo; //通过继承Thread类来创建线程类 public class MyThread extends Thread{ private int ticket = 10; //重写run方法 public void run(){ for(int i=0;i<20;i++){ if(this.ticket>0){ System.out.println(this.getName()+" 卖票:ticket"+this.ticket--); } } } public static void main(String[] args) { // 启动3个线程t1,t2,t3;每个线程各卖10张票! MyThread t1=new MyThread(); MyThread t2=new MyThread(); MyThread t3=new MyThread(); t1.start(); t2.start(); t3.start(); } }
运行结果:
Thread-0 卖票:ticket10 Thread-0 卖票:ticket9 Thread-0 卖票:ticket8 Thread-0 卖票:ticket7 Thread-1 卖票:ticket10 Thread-1 卖票:ticket9 Thread-1 卖票:ticket8 Thread-1 卖票:ticket7 Thread-1 卖票:ticket6 Thread-0 卖票:ticket6 Thread-0 卖票:ticket5 Thread-0 卖票:ticket4 Thread-0 卖票:ticket3 Thread-0 卖票:ticket2 Thread-0 卖票:ticket1 Thread-2 卖票:ticket10 Thread-2 卖票:ticket9 Thread-2 卖票:ticket8 Thread-2 卖票:ticket7 Thread-2 卖票:ticket6 Thread-2 卖票:ticket5 Thread-2 卖票:ticket4 Thread-2 卖票:ticket3 Thread-2 卖票:ticket2 Thread-1 卖票:ticket5 Thread-2 卖票:ticket1 Thread-1 卖票:ticket4 Thread-1 卖票:ticket3 Thread-1 卖票:ticket2 Thread-1 卖票:ticket1
结果说明:
(01) MyThread继承于Thread,它是自定义个线程。每个MyThread都会卖出10张票。
(02) 主线程main创建并启动3个MyThread子线程。每个子线程都各自卖出了10张票。
二、实现Runnable接口创建线程类
Runnable 是一个接口,该接口中只包含了一个run()方法。它的定义如下:
public interface Runnable { public abstract void run(); }
实现Runnable接口来创建并启动多线程的步骤如下:
01. 定义Runnable接口的实现类,并重写该接口的run()方法,该run()方法的方法体同样是该线程的线程执行体。
02. 创建Runnable实现类的实例,并以此实例作为Thread的target来创建Thread对象,该Thread对象才是真正的线程对象。
03. 调用线程对象的start()方法来启动线程。
需要注意的是:Runnable对象仅仅作为Thread对象的target,Runnable实现类里包含的run()方法仅作为线程执行体。而实际的线程对象依然是Thread实例,只是该Thread线程负责执行其target的run()方法。
示例:
package com.demo; //通过实现Runnable接口创建多线程 public class MyThread1 implements Runnable{ private int ticket = 10; public void run(){ for(int i=0;i<20;i++){ if(this.ticket>0){ System.out.println(Thread.currentThread().getName()+" 卖票:ticket"+this.ticket--); } } } public static void main(String[] args) { MyThread1 mt=new MyThread1(); // 启动3个线程t1,t2,t3(它们共用一个Runnable对象),这3个线程一共卖10张票! Thread t1=new Thread(mt); Thread t2=new Thread(mt); Thread t3=new Thread(mt); t1.start(); t2.start(); t3.start(); } }
运行结果:
Thread-0 卖票:ticket10 Thread-0 卖票:ticket8 Thread-0 卖票:ticket7 Thread-1 卖票:ticket9 Thread-2 卖票:ticket5 Thread-0 卖票:ticket6 Thread-1 卖票:ticket4 Thread-2 卖票:ticket3 Thread-1 卖票:ticket1 Thread-0 卖票:ticket2
结果说明:
(01) 和上面“MyThread继承于Thread”不同;这里的MyThread1实现了Thread接口。
(02) 主线程main创建并启动3个子线程,而且这3个子线程都是基于“mt这个Runnable对象”而创建的。运行结果是这3个子线程一共卖出了10张票。这说明它们是共享了MyThread接口的。
三、Thread和Runnable的异同点
Thread 和 Runnable 的相同点:都是“多线程的实现方式”。
Thread 和 Runnable 的不同点:
Thread 是类,而Runnable是接口;Thread本身是实现了Runnable接口的类。我们知道“一个类只能有一个父类,但是却能实现多个接口”,因此Runnable具有更好的扩展性。
此外,Runnable还可以用于“资源的共享”。即,多个线程都是基于某一个Runnable对象建立的,它们会共享Runnable对象上的资源。
通常,建议通过“Runnable”实现多线程!