-
从java的设计来看,通过继承
Thread
或者实现Runnable
接口来创建线程本质上没有区别,从jdk帮助文档可以看到Thread
类本身就实现了Runnable
接口 -
实现
Runnable
接口方式更加适合多个线程共享一个资源的情况,并且避免了单继承的限制,建议使用Runnable
。 -
【售票系统】,模拟三个售票窗口同时售票
100
张,分别使用继承Thread
和实现Runnable
方式,并分析有什么问题?
- 继承 Thread 方式实现,会出现“超卖”的问题
public class SellTicket {
public static void main(String[] args) {
SellTicket01 sellTicket01 = new SellTicket01();
SellTicket01 sellTicket02 = new SellTicket01();
SellTicket01 sellTicket03 = new SellTicket01();
//这里会出现票数超卖的现象
sellTicket01.start();
sellTicket02.start();
sellTicket03.start();
}
}
//使用继承 Thread 方式
class SellTicket01 extends Thread {
private static int ticketNum = 100; //让多个线程共享 ticketNum
@Override
public void run() {
while (true) {
if (ticketNum <= 0) {
System.out.println("售票结束...");
break;
}
//休眠50毫秒,模拟休息时间
try {
Thread.sleep(50);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("窗口 " + Thread.currentThread().getName() + " 售出一张票" + " 剩余票数" + (--ticketNum));
}
}
}
-
原因介绍如下:
当总票数最后只剩下 1 张时,三个窗口同时售卖一张票的情况下,就会导致票数变成 -2 张的情况 -
实现 Runnable 接口的方式来实现,就可以避免出现“超卖”的现象,因为这三个线程(窗口)指向同一个对象(sellTicket02)
public class SellTicket {
public static void main(String[] args) {
SellTicket02 sellTicket02 = new SellTicket02();
new Thread(sellTicket02).start(); //第1个线程-窗口
new Thread(sellTicket02).start(); //第2个线程-窗口
new Thread(sellTicket02).start(); //第3个线程-窗口
}
}
class SellTicket02 implements Runnable {
private int ticketNum = 100;
@Override
public void run() {
while (true) {
if (ticketNum <= 0) {
System.out.println("售票结束...");
break;
}
//休眠50毫秒,模拟休息时间
try {
Thread.sleep(50);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("窗口 " + Thread.currentThread().getName() + " 售出一张票" + " 剩余票数" + (--ticketNum));
}
}
}