美文网首页
ReentrantLock总结

ReentrantLock总结

作者: jackcooper | 来源:发表于2018-06-01 00:41 被阅读112次

    简述

    ReentrantLock是java中非常重要的一个并发工具,相比于java原生的synchronized有着更好的性能

    ## 概念速查
    ReentrantLock涉及的名称和概念较多,这里做一个简单的归类和解释,具体更为详细的内容,请自行Baidu或Google,这部分用于在阅读文章的时候,快速了解一些名称的概念,如果已经熟悉,请跳过。
    
    

    快速预览

    更强大的功能,玩玩意味着更为复杂的使用,ReentrankLock的使用比起synchronize,多了一个主动释放锁的代码,一个典型的使用示例如下

    ReentrantLock lock = new ReentrantLock();
    lock.lock();
    try {
        // ... method body
    }finally {
        lock.unlock();
    }
    
    

    注意unlock的操作一定要置于finally块中,这样才能保证锁一定能释放。

    uml图

    看完了简单的使用示例,我们来快速的看一遍ReentrankLock的结构,下面是用idea的工具快速生成的uml图,感谢idea,大大提高了我们的工作质量。

    ReentrantLock-UML

    一次性过于深入的讨论,往往会迷失在繁琐的细节中,而难以把握全貌,而细节往往是由全局的目标决定的,所以我们一层一层的谈,不一次性深入最终代码。

    由uml图,我们可以看出,ReentrantLock类是一个Lock接口的具体实现,每个ReentrantLock的实例,都持有一个sync对象,且这个sync是final修饰的,这个sync有两种具体的子类,分别是NonfairSync和FairSync,也就是非公平锁和公平锁。

    ReentrantLock有两个构造方法,我们可以先看这两个方法,

    public ReentrantLock() {
        sync = new NonfairSync();
    }
    public ReentrantLock(boolean fair) {
        sync = (fair)? new FairSync() : new NonfairSync();
    }
    
    

    可以看出,所谓构造函数,其实就是初始化需要使用的sync的类型,默认是非公平锁。参考公平锁与非公平锁

    案例:并发卖火车票

    public class Ticket implements Runnable {
    // 当前拥有的票数
    private int num = 100;
    ReentrantLock lock = new ReentrantLock();
    
    public void run() {
        while (true) {
            try {
                Thread.sleep(10);
            } catch (InterruptedException e) {
            }
            lock.lock();
            // 输出卖票信息
            if (num > 0) {
                System.out.println(Thread.currentThread().getName() + ".....sale...." + num--);
            }
            lock.unlock();
       }
    }
    }
    
    

    相关文章

      网友评论

          本文标题:ReentrantLock总结

          本文链接:https://www.haomeiwen.com/subject/kkthsftx.html