模拟卖票案例
创建三个线程,同时开启,对共享的票进行出售
注意:
线程安全问题是不能产生的,我们可以让一个线程在访问共享数据的时候,无论是否失去了cpu的执行权,
让其他的线程只能等待,等待当前线程买完票,其他线程在进行卖票
保证:使用一个线程在卖票
卖票案例出现了线程安全问日
卖出了不存在的票和重复的票
解决线程安全问题的第两种方案:使用同步方法
使用步骤:
1、把访问了共享数据的代码抽取出来,放到一个方法中
2、在方法上添加synchronization修饰符
public class RunnableImpl implements Runnable {
//定义一个多个线程共享的资源,一百张票
private static int ticket = 100;
@Override
public void run() {
//使用死循环,让卖票操作重复执行
System.out.println(this);//SynchronizationDemo02.RunnableImpl@1b6d3586
while (true){
//payTicket();
payTicketstatic();
}
}
/*
定义一个同步方法
同步方法也会把方法内部代码锁住
只让一个线程执行
同步方法的锁对象是谁?:就是实现类对象 new RunnableImpl()
也就是this
*/
public /*synchronized*/ void payTicket(){
synchronized (this){
if(ticket > 0){
//票存在,卖票 ticket--
try {
Thread.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"正在卖"+ticket+"票");
ticket--;
}
}
}
//提高安全问题出现的概率,让程序睡眠
//判断票是否存在
// if(ticket > 0){
// //票存在,卖票 ticket--
// try {
// Thread.sleep(10);
// } catch (InterruptedException e) {
// e.printStackTrace();
// }
// System.out.println(Thread.currentThread().getName()+"正在卖"+ticket+"票");
// ticket--;
// }
// }
/*
静态同步方法
锁对象不能是this
this是创建对象之后产生的,静态方法优先于对象
静态对象的锁对象是本类的class属性-->class文件对象(反射)
*/
public static /*synchronized*/ void payTicketstatic(){
synchronized (RunnableImpl.class){
//提高安全问题出现的概率,让程序睡眠
//判断票是否存在
if(ticket > 0){
//票存在,卖票 ticket--
try {
Thread.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"正在卖"+ticket+"票");
ticket--;
}
}
}
}
public class ThreadDemo03 {
public static void main(String[] args) {
//创建Runnable接口实现类对象
RunnableImpl run = new RunnableImpl();
System.out.println("-------"+run);//-------SynchronizationDemo02.RunnableImpl@1b6d3586
//创建Thread类对象,构造方法中传递Runnable接口的实现类对象
Thread t1 = new Thread(run);
Thread t2 = new Thread(run);
Thread t3 = new Thread(run);
//开启多线程
t1.start();
t2.start();
t3.start();
}
}