最近学习遇到了经典的多线程并发的卖票问题,在网上搜了一些答案,还是决定自己写一写看,于是就出现了这篇文章,希望对初学者有些帮助!!!
package practice.threadsafe;
/*
多线程并发卖票操作
*/
public class BuyTicket {
public static void main(String[] args) {
Tickets t=new Tickets(50);
//创建两个线程对象
Thread t1=new SellTicket(t);
Thread t2=new SellTicket(t);
//修改线程名
t1.setName("t1");
t2.setName("t2");
//线程启动
t1.start();
t2.start();
}
}
class Tickets{
//set和get方法一定要写,对number进行封装
private int number;
public Tickets(){
}
public Tickets(int number){
this.number=number;
}
public void setNumber(int number){
this.number=number;
}
public int getNumber(){
return number;
}
public void buy(){
//线程是需要调用buy()方法来进行购票操作,所以一次只能一个线程调用该方法
//所以buy()内代码块要加上synchronized关键字修饰
//this指的是当前对象,即Tickets,因为它是两个线程的共享对象
synchronized (this){
if (this.getNumber() >= 1) {
System.out.println(Thread.currentThread().getName() + "号窗口卖出一张票,剩余总票数为" + (this.getNumber() - 1) + "张");
}
//让线程睡眠0.1秒
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
int num = this.getNumber();
num--;
this.setNumber(num);
}
}
}
class SellTicket extends Thread {
private Tickets tickets;
public SellTicket(Tickets tickets) {
this.tickets = tickets;
}
@Override
public void run() {
//执行卖票操作
while (true) {
if (tickets.getNumber() > 0) {
tickets.buy();
} else {
//当余票为0时,就退出循环,卖票结束
break;
}
}
}
}
问题解决:一开始我将余票的输出写在了run()方法中,但后来发现不妥,因为余票一个和buy方法绑在一起,所以我将余票输出的代码添加在了buy()方法当中,这样子buy()方法调用一次,余票就输出一次,不会出错!
最后运行结果如下:
ps:这t2线程也太强了
t2号窗口卖出一张票,剩余总票数为49张
t2号窗口卖出一张票,剩余总票数为48张
t2号窗口卖出一张票,剩余总票数为47张
t2号窗口卖出一张票,剩余总票数为46张
t2号窗口卖出一张票,剩余总票数为45张
t2号窗口卖出一张票,剩余总票数为44张
t2号窗口卖出一张票,剩余总票数为43张
t2号窗口卖出一张票,剩余总票数为42张
t2号窗口卖出一张票,剩余总票数为41张
t2号窗口卖出一张票,剩余总票数为40张
t2号窗口卖出一张票,剩余总票数为39张
t1号窗口卖出一张票,剩余总票数为38张
t2号窗口卖出一张票,剩余总票数为37张
t1号窗口卖出一张票,剩余总票数为36张
t1号窗口卖出一张票,剩余总票数为35张
t2号窗口卖出一张票,剩余总票数为34张
t2号窗口卖出一张票,剩余总票数为33张
t2号窗口卖出一张票,剩余总票数为32张
t2号窗口卖出一张票,剩余总票数为31张
t2号窗口卖出一张票,剩余总票数为30张
t1号窗口卖出一张票,剩余总票数为29张
t1号窗口卖出一张票,剩余总票数为28张
t1号窗口卖出一张票,剩余总票数为27张
t1号窗口卖出一张票,剩余总票数为26张
t1号窗口卖出一张票,剩余总票数为25张
t1号窗口卖出一张票,剩余总票数为24张
t1号窗口卖出一张票,剩余总票数为23张
t2号窗口卖出一张票,剩余总票数为22张
t2号窗口卖出一张票,剩余总票数为21张
t2号窗口卖出一张票,剩余总票数为20张
t2号窗口卖出一张票,剩余总票数为19张
t2号窗口卖出一张票,剩余总票数为18张
t2号窗口卖出一张票,剩余总票数为17张
t2号窗口卖出一张票,剩余总票数为16张
t1号窗口卖出一张票,剩余总票数为15张
t2号窗口卖出一张票,剩余总票数为14张
t2号窗口卖出一张票,剩余总票数为13张
t2号窗口卖出一张票,剩余总票数为12张
t2号窗口卖出一张票,剩余总票数为11张
t2号窗口卖出一张票,剩余总票数为10张
t2号窗口卖出一张票,剩余总票数为9张
t2号窗口卖出一张票,剩余总票数为8张
t2号窗口卖出一张票,剩余总票数为7张
t2号窗口卖出一张票,剩余总票数为6张
t2号窗口卖出一张票,剩余总票数为5张
t1号窗口卖出一张票,剩余总票数为4张
t2号窗口卖出一张票,剩余总票数为3张
t2号窗口卖出一张票,剩余总票数为2张
t2号窗口卖出一张票,剩余总票数为1张
t2号窗口卖出一张票,剩余总票数为0张
进程已结束,退出代码 0
|