> 文档中心 > 案例:卖票(含原因分析)

案例:卖票(含原因分析)


案例:卖票

  • 需求:某电影院目前正在上映国产大片,共有100张票,而他有3个窗口卖票,请设计一个程序模拟该电影卖票

  • 思路:

  1. 定义一个SellTicket类实现Runnable接口,里面定义一个成员变量:private int tickets=100;

  2. 在SellTicket类中重写run()方法实现卖票,代码步骤如下:
    A:判断票数大于0,就卖票,并告知是哪个窗口卖的
    B:卖了票之后,总票数减1
    C:票没有了,也可能有人来问,所以这里用死循环让卖票动作一直执行

  3. 定义一个测试类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();    }}

注意问题:

  • 卖票出现的问题:
    相同的票出现了多次
    出现了负数的票
  • 问题的原因:
    线程的随机性导致的