模拟卖票案例
创建三个线程,同时开启,对共享的票进行出售
注意:
线程安全问题是不能产生的,我们可以让一个线程在访问共享数据的时候,无论是否失去了cpu的执行权,
让其他的线程只能等待,等待当前线程买完票,其他线程在进行卖票
保证:使用一个线程在卖票
卖票案例出现了线程安全问日
卖出了不存在的票和重复的票
解决线程安全问题的第三种方案:使用Lock锁
使用步骤:
1、在成员位置创建一个ReentrantLock对象
2、在可能会出现安全问题的代码前调用Lock接口中的方法Lock获取锁
3、在可能会出现安全问题的代码后调用Lock接口中的方法unLock释放锁
public class RunnableImpl implements Runnable {
//定义一个多个线程共享的资源,一百张票
private int ticket = 100;
//在成员位置创建一个ReentrantLock对象
ReentrantLock l = new ReentrantLock();
@Override
// public void run() {
// while (true){
// l.lock();
// if(ticket > 0){
// //票存在,卖票 ticket--
// try {
// Thread.sleep(10);
// } catch (InterruptedException e) {
// e.printStackTrace();
// }
// System.out.println(Thread.currentThread().getName()+"正在卖"+ticket+"票");
// ticket--;
// }
// l.unlock();
// }
// }
public void run() {
while (true){
l.lock();
if(ticket > 0){
//票存在,卖票 ticket--
try {
Thread.sleep(10);
System.out.println(Thread.currentThread().getName()+"正在卖"+ticket+"票");
ticket--;
} catch (InterruptedException e) {
e.printStackTrace();
}finally {
//无论代码是否出现异常,锁总会释放掉
l.unlock();
}
}
}
}
}
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();
}
}