JAVA线程经典案例 多线程买票 详细解析
很经典的多线程题目,根据自己的理解得出的解析过程和结论,有不对的地方希望指出。
需求:
建立购票系统,分别让多个窗口(多线程)同时去卖票
-
创建个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--; } } } } }
上一篇:
IDEA上Java项目控制台中文乱码