很经典的多线程题目,根据自己的理解得出的解析过程和结论,有不对的地方希望指出。
需求:
建立购票系统,分别让多个窗口(多线程)同时去卖票
-
创建个Runnable接口的实现类
-
重写其中的run()方法,变成购票方法
-
创建多个接口类对象(即多个窗口)放入线程中多个执行
package Chating;
public class Maipiao {
public static void main(String[] args) {
per per = new per();//创建线程实现类的对象
Thread t1 = new Thread(per);//注意只能new一个类放入Thread()中启动,保证同用10张票这一个资源
Thread t2 = new Thread(per);
Thread t3 = new Thread(per);
t1.start();//启动线程
t2.start();
t3.start();
}
}
class per implements Runnable {//创建Runnable接口的实现类
int ticket = 10;//一共要出售10张票
@Override
public void run() {//重写run()方法
while (true) {
if (ticket > 0) {//当剩余票大于0张时执行
System.out.println(Thread.currentThread().getName() + "正在卖第" + ticket + "张票");//获取线程名 打印卖票信息
ticket--;//每出售一张票后数量自动-1
}
}
}
}
能看到这里实际是有问题的,Thread-1 Thread-2 同卖第9张票,Thread-2 在卖第0张票,Thread-1 卖到第-1张,这就是线程的安全问题
问题原理分析
窗口卖掉了重复的票和不存在的票
Thread-1 Thread-2 同卖第9张票:
- 两条线程随机抢票,即正好同时抢到一张票
Thread-2 在卖第0张票,Thread-1 卖到第-1张:
- Thread-0,1,2三条线程正好同时抢到最后一张,线程0率先执行,卖掉票后-1剩余的票数为0。线程2拿掉的票就变成第0张了,无法满足if (ticket > 0)的条件,这条线程就停止了。因为线程2又拿掉了一张第0,最后执行的线程1手中的票变成了第-1张,同样无法满足if (ticket > 0)的条件,这条线程也停了。至此三条线程全部停止,打印出来的结果就如图所示了。
解决方法
可以利用同步代码块,锁住线程,只让一个线程在同步代码块执行,等该线程执行完毕后,下一个线程再继续进来执行
-
创建个Runnable接口的实现类
-
重写其中的run()方法,变成购票方法
-
创建同步代码块的锁对象,该对象可以任意,但必须保证代码块中锁对象始终是唯一不变的
-
在容易出问题的代码上套入同步代码块将其锁住
-
创建多个接口类对象(即多个窗口)放入线程中多个执行
package Chating;
public class Maipiao2 {
public static void main(String[] args) {
per per = new per();
Thread t1 = new Thread(per);
Thread t2 = new Thread(per);
Thread t3 = new Thread(per);
t1.start();
t2.start();
t3.start();
}
}
class per implements Runnable{
int ticket = 10;
final String a = "";//任意定义的锁对象,只要保证唯一不变即可
@Override
public void run() {
while (true) {
synchronized(a) {//同步代码块,a是锁对象
if (ticket>0) {
try {
Thread.sleep(100);//sleep让程序休眠,加大出现问题的概率,方便定位问题,使用时会有异常try..catch即可
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"正在卖第"+ticket+"张票");
ticket--;
}
}
}
}
}