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--;
				}
			}
		}
	}
}

在这里插入图片描述

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值