模拟卖票案例
创建三个线程,同时开启,对共享的票进行出售
注意:
线程安全问题是不能产生的,我们可以让一个线程在访问共享数据的时候,无论是否失去了cpu的执行权,
让其他的线程只能等待,等待当前线程买完票,其他线程在进行卖票
保证:使用一个线程在卖票
卖票案例出现了线程安全问题
卖出了不存在的票和重复的票
解决线程安全问题的一种方案:使用同步代码块
格式:
Synchronization(锁对象){
可能会出现线程安全问题的代码(访问了共享资源的代码)
}
注意:
1、通过代码块中的锁对象,可以使用任意的对象
2、必须保证多个线程使用的锁对象是同一个
3、锁对象的作用:
把同步代码块锁住,只让一个线程在同步代码块执行
public class RunnableImpl implements Runnable {
//定义一个多个线程共享的资源,一百张票
private int ticket = 100;
//创建一个锁对象,必须创建在run方法外面,不然创建三个线程就有三个锁
Object obj = new Object();
@Override
public void run() {
//使用死循环,让卖票操作重复执行
while (true){
//创建一个同步代码块
synchronized (obj){
//提高安全问题出现的概率,让程序睡眠
//判断票是否存在
if(ticket > 0){
//票存在,卖票 ticket--
System.out.println(Thread.currentThread().getName()+"正在卖"+ticket+"票");
ticket--;
}
}
}
}
}
public class ThreadDemo03 {
public static void main(String[] args) {
//创建Runnable接口实现类对象
RunnableImpl run = new RunnableImpl();
//创建Thread类对象,构造方法中传递Runnable接口的实现类对象
Thread t1 = new Thread(run);
Thread t2 = new Thread(run);
Thread t3 = new Thread(run);
//开启多线程
t1.start();
t2.start();
t3.start();
}
}