Java 多线程(六):ReentrantLock 与 Condition

ReentrantLock

  • java.util.concurrent.lock 中的 Lock 是锁的顶层接口,它允许把锁定的实现作为 Java 类,而不是作为语言特性来实现,这带来了更多的灵活性,可以只对某个代码块进行加锁,而不是整个方法,ReentrantLock 是 Lock 的一种实现
  • 它有几个主要的方法:
public interface Lock {
    void lock();//获得锁,如果锁已经被占用则等待,必须等待当前线程结束才会响应其它线程的中断
    void lockInterruptibly() throws InterruptedException;//获得锁,但如果检测到 interrupt 标志为 true 则立刻抛出 InterruptedException 异常
    boolean tryLock();// 尝试获得锁,如果成功,返回true,失败返回false。该方法不等待,立即返回
    boolean tryLock(long time, TimeUnit unit) throws InterruptedException; //   在给定时间内尝试获得锁
    void unlock();   //用来释放锁
    Condition newCondition();
}
  • 注意:解锁操作要放到 finally 中,这样保证即使抛出了异常锁也必须释放,否则其它线程将永远阻塞

用法

  • 下面例子中,线程2必须等待线程1释放锁后才能执行锁内的代码
public class LockTest implements Runnable{

    public static ReentrantLock reentrantLock = new ReentrantLock();

    @Override
    public void run() {
        System.out.println(Thread.currentThread().getId()+"进入方法");
        try {
            // reentrantLock.lockInterruptibly();  检测到 interrupt 标志则立刻抛出异常
            reentrantLock.lock();
            Thread.sleep(2000);
            System.out.println(Thread.currentThread().getId()+"执行完成");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }finally {
            reentrantLock.unlock();
        }
    }

    public static void main(String[] args) {
        LockTest lockTest = new LockTest();
        new Thread(lockTest).start();
        new Thread(lockTest).start();
    }

}

公平锁和非公平锁

  • 公平锁是指多个线程等待同一个锁时,必须按照申请锁的先后顺序来获得锁
  • 非公平锁是指可以不按照顺序,抢占锁
// 构造参数为 true 为公平锁(默认),false 为非公平锁
public static ReentrantLock reentrantLock = new ReentrantLock( false);

Condition

  • Condition 的功能与 wait() 和 notify() 方法差不多,但前者是配合 ReentrantLock 使用的,后者是配合 synchronized 使用的
  • 它几个主要的方法:
void await() throws InterruptedException; // 使当前线程等待,同时释放锁,当其它线程中使用 singal() 和 signalAll() 方法时,线程会重新获得锁并继续执行,或者当前线程中断时也跳出等待
void awaitUninterruptibly();    // 与 await() 方法基本相同,但不会响应中断
void signal();  // 唤醒一个在等待的线程
void signalAll();   // 唤醒所有在等待的线程

用法

  • 下面代码中线程1会进入等待状态,并释放锁,直到线程2调用 signal() 方法重新唤醒时才继续执行
public class LockTest implements Runnable{

    public static ReentrantLock reentrantLock = new ReentrantLock( );
    public static Condition condition = reentrantLock.newCondition();

    public Boolean flag;

    public LockTest(Boolean flag) {
        this.flag = flag;
    }

    @Override
    public void run() {
        System.out.println(Thread.currentThread().getId()+"进入方法");
        try {
            reentrantLock.lock();
            System.out.println(Thread.currentThread().getId()+"执行中");
            if(flag)
                condition.signal();
            else
                condition.await();
            System.out.println(Thread.currentThread().getId()+"执行完成");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }finally {
            reentrantLock.unlock();
        }
    }

    public static void main(String[] args) {
        LockTest l1 = new LockTest(false)
        new Thread(l1).start();
        LockTest l2 = new LockTest(true);
        new Thread(l2).start();
    }

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • Java-Review-Note——4.多线程 标签: JavaStudy PS:本来是分开三篇的,后来想想还是整...
    coder_pig阅读 1,750评论 2 17
  • 线程安全 当多个线程访问一个对象时,如果不用考虑这些线程在运行时环境下的调度和交替执行,也不需要进行额外的同步,或...
    闽越布衣阅读 803评论 0 6
  • 进程和线程 进程 所有运行中的任务通常对应一个进程,当一个程序进入内存运行时,即变成一个进程.进程是处于运行过程中...
    小徐andorid阅读 2,963评论 3 53
  • 2017年8月17日种种子 何德胜第17天 我今天不是为了我一个人而闻思修行,而是为了一切如母有情众生,早日离苦得...
    何德胜觉悟阅读 184评论 0 3
  • 寒假阅读完好几本书,文章数却没有增加,只有硬逼自己写的《枢纽》笔记,我不禁为年度30篇文章的目标隐隐担忧,唯有不断...
    罗秋秋啾啾啾啾啾阅读 576评论 6 5

友情链接更多精彩内容