案例:卖票
-
需求:某电影院目前正在上映国产大片,共有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();
}
}
注意问题:
- 卖票出现的问题:
相同的票出现了多次
出现了负数的票 - 问题的原因:
线程的随机性导致的