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张票@Overridepublic 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 = "";//任意定义的锁对象,只要保证唯一不变即可@Overridepublic void run() {while (true) {synchronized(a) {//同步代码块,a是锁对象if (ticket>0) {try {Thread.sleep(100);//sleep让程序休眠,加大出现问题的概率,方便定位问题,使用时会有异常try..catch即可} catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace();}System.out.println(Thread.currentThread().getName()+"正在卖第"+ticket+"张票");ticket--;}}}}
}

本文来自互联网用户投稿,文章观点仅代表作者本人,不代表本站立场,不承担相关法律责任。如若转载,请注明出处。 如若内容造成侵权/违法违规/事实不符,请点击【内容举报】进行投诉反馈!
