卖票案例:解决线程安全问题,Lock锁

package com.chunzhi.Test09Lock;
/*
    模拟卖票案例
    创建3个线程,同时开启,对共享的票进行出售
 */
public class Test01Ticket {
    public static void main(String[] args) {
        // 创建Runnable接口的实现类对象
        RunnableImpl3 run = new RunnableImpl3();
        // 创建Thread类对象,构造方法中传递Runnable接口的实现类对象
        Thread t1 = new Thread(run);
        Thread t2 = new Thread(run);
        Thread t3 = new Thread(run);
        // 调用start方法开启多线程
        t1.start();
        t2.start();
        t3.start();
    }
}
package com.chunzhi.Test09Lock;

import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

/*
    解决线程安全问题的第三种方案:使用Lock锁
    java.util.concurrent.locks.Lock接口
    Lock实现提供了比使用synchronized方法和语句可获得的更广泛的锁定操作。
    Lock接口中的方法:
        void lock()获取锁
        void unlock() 释放锁
    java.util.concurrent.locks.ReentrantLock implements Lock接口

    使用步骤:
        1.在成员位置创建一个ReentrantLock对象
        2.在可能会出现安全问题的代码前调用Lock接口中的方法lock获取锁
        3.在可能会出现安全问题的代码后调用Lock接口中的方法unlock释放锁
 */
public class RunnableImpl3 implements Runnable {
    // 定义一个多线程共享的票源
    private int ticket = 100; // 现在多核多线程计算的CPU太快了,这里可以适当调更大的值

    // 1.在成员位置创建一个ReentrantLock对象
    Lock l = new ReentrantLock();

    // 设置线程任务:卖票
    @Override
    public void run() {
        // 使用死循环,让卖票操作重复执行
        while (true) {
            // 2.在可能会出现安全问题的代码前调用Lock接口中的方法lock获取锁
            l.lock();
            // 先判断票是否存在
            if (ticket > 0) {
                // 有票,卖票
                System.out.println(Thread.currentThread().getName() + "正在卖第" + ticket + "张票");
                // 卖一次票减一张票
                ticket--;
            }
            // 3.在可能会出现安全问题的代码后调用Lock接口中的方法unlock释放锁
            l.unlock();
        }
    }
}

 

posted @ 2020-11-13 16:48  春志  阅读(192)  评论(0)    收藏  举报