案例:卖票(含原因分析)
案例:卖票
-
需求:某电影院目前正在上映国产大片,共有100张票,而他有3个窗口卖票,请设计一个程序模拟该电影卖票
-
思路:
-
定义一个SellTicket类实现Runnable接口,里面定义一个成员变量:private int tickets=100;
-
在SellTicket类中重写run()方法实现卖票,代码步骤如下:
A:判断票数大于0,就卖票,并告知是哪个窗口卖的
B:卖了票之后,总票数减1
C:票没有了,也可能有人来问,所以这里用死循环让卖票动作一直执行 -
定义一个测试类SellTicketDemo,里面有main方法,代码步骤如下:
A:创建SellTicket类的对象
B:创建3个Thread类的对象,把SellTicket对象作为构造方法的参数,并给对应的窗口名称
C:启动线程
以代码内容形式讲解
SellTicKet类(注意了)
内部含有原因分析
package Demo;public class SellTicket implements Runnable { private int tickets = 100; @Override public void run() {// while(true){ //相同票的情况 //tickets=100; //t1,t2,t3 //假设t1线程抢到CPU执行权// if (tickets>0){// //通过sleep()方法来模拟时间// try {// Thread.sleep(100);// //t1线程休息100秒// //t2线程抢到了CPU的执行权,t2线程就开始执行,执行到这里的时候,t2线程休息100秒// //t3线程抢到了CPU的执行权,t3线程就开始执行,执行到这里的时候,t3线程休息100秒// } catch (InterruptedException e) {// e.printStackTrace();// }// //假设线程按照顺序醒过来// //t1抢到CPU的执行权在控制台输出,窗口1正在出售第100张票// System.out.println("在"+Thread.currentThread().getName()+"号卖的票第"+tickets+"票"");// //此时这时t2抢到CPU的执行权在控制台输出,窗口2正在出售第100张票// //此时又,t3抢到CPU的执行权在控制台输出,窗口3正在出售第100张票// tickets--;// //如果这三个线程还是按照顺序来,这里就执行了3次--的操作,最终票成为97// }// } //出现负数票的情况 while (true) { //tickets=1; //t1,t2,t3 //假设t1线程抢到CPU执行权 if (tickets > 0) { //通过sleep()方法来模拟时间 try { Thread.sleep(100); //t1线程休息100秒 //t2线程抢到了CPU的执行权,t2线程就开始执行,执行到这里的时候,t2线程休息100秒 //t3线程抢到了CPU的执行权,t3线程就开始执行,执行到这里的时候,t3线程休息100秒 } catch (InterruptedException e) { e.printStackTrace(); } //假设线程按照顺序醒过来 //t1抢到CPU的执行权在控制台输出,窗口1正在出售第1张票 System.out.println("在" + Thread.currentThread().getName() + "号卖的票第" + tickets + "票"); //假设t1继续拥有CPU的执行权,就会执行tickets--;操作。 此时tickets=0; //即使t1还是拥有CPU执行权,但是不符合if条件,注意了这时t2不通过if条件的限制 //t2抢到CPU的执行权在控制台输出,窗口2正在出售第0张票 (注意这个话的意思) //假设t2继续拥有CPU的执行权,就会执行tickets--;操作。 此时tickets=-1; (注意啦) //注意了这时t2不通过if条件的限制,t3抢到CPU的执行权在控制台输出,窗口3正在出售第-1张票 (注意这个话的意思) //假设t3继续拥有CPU的执行权,就会执行tickets--;操作。 此时tickets=-2; (注意啦) tickets--; } } } }
SellTicKetDemo类
package Demo;public class SellTicketDemo { public static void main(String[] args) { //创建SellTicket类的对象 Runnable st = new SellTicket(); //创建Thread类的对象,把SellTicket对象作为构造方法的参数 Thread tr = new Thread(st,"窗口1"); Thread tr2 = new Thread(st,"窗口2"); Thread tr3 = new Thread(st,"窗口3"); //启动线程 tr.start(); tr2.start(); tr3.start(); }}
注意问题:
- 卖票出现的问题:
相同的票出现了多次
出现了负数的票 - 问题的原因:
线程的随机性导致的