1.1 进程和线程关系
1.2 线程的关系
线程图
1.3 实现Runnable接口相对于继承Thread类来说,有如下显著的好处:
(1)适合多个相同程序代码的线程去处理同一资源的情况,把虚拟CPU(线程)同程序的代码,数据有效的分离,较好地体现了面向对象的设计思想
(2)可以避免由于Java的单继承特性带来的局限。我们经常碰到这样一种情况,即当我们要将已经继承了某一个类的子类放入多线程中,由于一个类不能同时有两个父类,所以不能用继承Thread类的方式,那么,这个类就只能采用实现Runnable接口的方式了。
(3)有利于程序的健壮性,代码能够被多个线程共享,代码与数据是独立的。当多个线程的执行代码来自同一个类的实例时,即称它们共享相同的代码。多个线程操作相同的数据,与它们的代码无关。当共享访问相同的对象是,即它们共享相同的数据。当线程被构造时,需要的代码和数据通过一个对象作为构造函数实参传递进去,这个对象就是一个实现了Runnable接口的类的实例。
具体场景如:实现通过四个售票点发售某日某次列车的100张车票,一个售票点用一个线程表示。
package com.base.zhou.thread.pack2;
/**
* @author zhou.moumou
* @version 2018年5月7日 下午8:15:54
* @description
*/
public class ThreadTest {
public static void main(String[] args) {
ThreadTest2 t = new ThreadTest2();
t.start();
t.start();
t.start();
t.start();
}
}
class ThreadTest2 extends Thread {
private int ticket = 100;
public void run() {
while (true) {
if (ticket > 0) {
System.out.println(Thread.currentThread().getName() + "is saling ticket" + ticket--);
} else {
break;
}
}
}
程序运行结果:
上面的代码中,我们用ThreadTest2类模拟售票处的售票过程,run方法中的每一次循环都将总票数减1,模拟卖出一张车票,同时该车票号打印出来,直接剩余的票数到零为止。在ThreadTest类的main方法中,我们创建了一个线程对象,并重复启动四次,希望通过这种方式产生四个线程。从运行的结果来看我们发现其实只有一个线程在运行,这个结果 告诉我们:第一: 一个线程对象只能启动一个线程,无论你调用多少遍start()方法,结果只有一个线程。第二: 为何会报这个错误,原因就是从new到等待运行是单行道,所以如果你对一个已经启动的线程对象再调用一次start方法的话,会产生:IllegalThreadStateException异常
如果你还是不信邪,说可以启动四个ThreadTest2来start,那么ticket就不是共享的了,具体代码不加演示
上述解决方式就是使用Runnable了
代码:
package com.base.zhou.thread.pack2;
/**
* @author zhou.moumou
* @version 2018年5月7日 下午8:15:54
* @description
*/
public class ThreadTest {
public static void main(String[] args) {
/*
* ThreadTest2 t = new ThreadTest2(); t.start(); t.start(); t.start();
* t.start();
*/
ThreadTest3 t = new ThreadTest3();
new Thread(t).start();
new Thread(t).start();
new Thread(t).start();
new Thread(t).start();
}
}
class ThreadTest3 implements Runnable {
private int tickets = 100;
public void run() {
while (true) {
if (tickets > 0) {
System.out.println(Thread.currentThread().getName() + " is saling ticket " + tickets--);
}
}
}
}
class ThreadTest2 extends Thread {
private int ticket = 100;
public void run() {
while (true) {
if (ticket > 0) {
System.out.println(Thread.currentThread().getName() + "is saling ticket" + ticket--);
} else {
break;
}
}
}
}
运行结果