前言 
最近开始读JDK源码,所有心得准备总结成一个专栏,JDK Analysis系列的第一篇,就从万众瞩目的ReentrantLock开始吧,而谈到ReentrantLock,就不得不说AQS,它是AbstractQueuedSynchronizer类的简称,Doug Lea上神在JDK1.5将其引入,这才有了现在的并发包java.util.concurrent,所以要理解ReentrantLock的原理,AQS也是必须要搞懂的。这篇就先阐述ReentrantLock最基本的公平锁和非公平锁的实现,以及部分涉及的AQS原理,AQS源码解读将在后续跟进。整个系列基于JDK1.8.0_92。

公平锁与非公平锁 
大家都知道,在JDK1.5之前,我们在多线程的环境下要想保证线程安全,就必须要使用synchronized关键字来实现对象锁或者类锁,以此满足这样的需求,JDK1.5之后则使用Lock来实现更加细粒度的锁。在刚接触Java的时候,学到这两种方式的时候,粗略地知道后者更加贴近面向对象的思想,但是在工作中遇到一些奇奇怪怪的需求的时候,只是知道这个是远远不够的。所谓公平锁,就是线程按照执行顺序排成一排,依次获取锁,但是这种方式在高并发的场景下极其损耗性能;这时候,非公平锁应运而生了,所谓非公平锁,就是不管执行顺序,每个线程获取锁的几率都是相同的,获取失败了,才会采用像公平锁那样的方式。这样做的好处是,JVM可以花比较少的时间在线程调度上,更多的时间则是用在执行逻辑代码里面。

公平锁、非公平锁的创建方式:

//创建一个非公平锁,默认是非公平锁
Lock lock = new ReentrantLock();
Lock lock = new ReentrantLock(false);

//创建一个公平锁,构造传参true
Lock lock = new ReentrantLock(true);
1
2
3
4
5
6
相关源码:

public ReentrantLock() {
        sync = new NonfairSync();
    }

public ReentrantLock(boolean fair) {
        sync = fair ? new FairSync() : new NonfairSync();
    }
1
2
3
4
5
6
7
这部分源码比较简单,这里对于源码就不赘述。

NonfairSync 非公平锁 
在谈NonfairSync之前,首先要谈谈ReentrantLock类里面定义的一个类属性Sync,它才是ReentrantLock实现的精髓。它首先在属性里声明,然后以抽象静态内部类的形式实现了AQS,源码如下:

abstract static class Sync extends AbstractQueuedSynchronizer {
        private static final long serialVersionUID = -5179523762034025860L;

//声明的lock()方法,供子类实现
        abstract void lock();

//非公平锁的获取方式,相较于公平锁的tryAcquire()
        final boolean nonfairTryAcquire(int acquires) {
            final Thread current = Thread.currentThread();
            int c = getState();
            if (c == 0) {
                if (compareAndSetState(0, acquires)) {
                    setExclusiveOwnerThread(current);
                    return true;
                }
            }
            else if (current == getExclusiveOwnerThread()) {
                int nextc = c + acquires;
                if (nextc < 0) // overflow
                    throw new Error("Maximum lock count exceeded");
                setState(nextc);
                return true;
            }
            return false;
        }

//释放锁
        protected final boolean tryRelease(int releases) {
            int c = getState() - releases;
            if (Thread.currentThread() != getExclusiveOwnerThread())
                throw new IllegalMonitorStateException();
            boolean free = false;
            if (c == 0) {
                free = true;
                setExclusiveOwnerThread(null);
            }
            setState(c);
            return free;
        }

//判断当前线程是否是锁的持有者
        protected final boolean isHeldExclusively() {
            return getExclusiveOwnerThread() == Thread.currentThread();
        }

final ConditionObject newCondition() {
            return new ConditionObject();
        }

//获取当前锁持有线程,如果在队列中等待获取锁,则返回null
        final Thread getOwner() {
            return getState() == 0 ? null : getExclusiveOwnerThread();
        }

//返回当前线程status的状态,如果持有锁就读取status,没有就0
        final int getHoldCount() {
            return isHeldExclusively() ? getState() : 0;
        }

//是否上锁
        final boolean isLocked() {
            return getState() != 0;
        }

/**
         * Reconstitutes the instance from a stream (that is, deserializes it).
         */
        private void readObject(java.io.ObjectInputStream s)
            throws java.io.IOException, ClassNotFoundException {
            s.defaultReadObject();
            setState(0); // reset to unlocked state
        }
    }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
此后,NonfairSync继承它来实现非公平锁,FairSync继承它来实现公平锁,AQS提供一个tryAcquire()的模板方法来使得公平锁和非公平锁的实现方式显得灵活。我们来看看NonfairSync的源码:

static final class NonfairSync extends Sync {
        private static final long serialVersionUID = 7316153563782823691L;

final void lock() {
            if (compareAndSetState(0, 1))
                setExclusiveOwnerThread(Thread.currentThread());
            else
                acquire(1);
        }

protected final boolean tryAcquire(int acquires) {
            return nonfairTryAcquire(acquires);
        }
    }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
如代码所示,在lock的时候,先是尝试将AQS的status从0设为1,成功的话就把当前线程设置为锁的持有者,如果尝试失败了,基于模板方法,实际调用的是Sync的nonfairTryAcquire(int acquires)方法,该方法源码如下:

final boolean nonfairTryAcquire(int acquires) {
            final Thread current = Thread.currentThread();
            int c = getState();
            if (c == 0) {
                if (compareAndSetState(0, acquires)) {
                    setExclusiveOwnerThread(current);
                    return true;
                }
            }
            //ReentrantLock是可重入锁是这里实现的
            else if (current == getExclusiveOwnerThread()) {
                int nextc = c + acquires;
                if (nextc < 0) // overflow
                    throw new Error("Maximum lock count exceeded");
                setState(nextc);
                return true;
            }
            return false;
        }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
首先获取当前线程,和当前AQS维护的锁的状态,如果状态为0,则尝试将AQS的status从0设为acquires(实际是1),如果设置成功,则获取锁成功,把当前锁设置为锁的持有者,返回true;如果当前线程已经是锁的持有者,则把status+acquires,如果结果越界,抛出异常,如果成功,返回true。细心的同学可以发现,一共有两次原子设status从0到1,为什么呢?因为这样可以提高获取锁的概率,因为是非公平的,所以有必要进行这样的操作,而且这样的操作与锁相对来讲损耗微乎其微。

FairSync 公平锁 
公平锁就是每个线程在获取锁时会先查看此锁维护的等待队列,如果为空,或者当前线程线程是等待队列的第一个,就占有锁,否则就会加入到等待队列中,以后会按照FIFO的规则从队列中获取,下面是FairSync 的源码:

static final class FairSync extends Sync {
        private static final long serialVersionUID = -3000897897090466540L;

final void lock() {
            acquire(1);
        }

protected final boolean tryAcquire(int acquires) {
            final Thread current = Thread.currentThread();
            int c = getState();
            if (c == 0) {
                if (!hasQueuedPredecessors() &&
                    compareAndSetState(0, acquires)) {
                    setExclusiveOwnerThread(current);
                    return true;
                }
            }
            else if (current == getExclusiveOwnerThread()) {
                int nextc = c + acquires;
                if (nextc < 0)
                    throw new Error("Maximum lock count exceeded");
                setState(nextc);
                return true;
            }
            return false;
        }
    }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
来看子类的 tryAcquire方法,与非公平锁比较,获取锁的操作只有一点不同,就是加入了hasQueuedPredecessors() 方法,该方法又大有来头,ctrl进去是这的:

public final boolean hasQueuedPredecessors() {
        Node t = tail; 
        Node h = head;
        Node s;
        //head没有next ----> false
        //head有next,next持有的线程不是当前线程 ----> true
        //head有next,next持有的线程是当前线程 ----> false
        return h != t && ((s = h.next) == null || s.thread != Thread.currentThread());
    }
1
2
3
4
5
6
7
8
9
该方法的签名是:查询是否有其他线程比当前线程等待获取锁花费了更多的时间。在AQS中对线程是做了一个FIFO队列,这里的tail是尾,head是头,具体的实现会在后续跟进,这里就不多做赘述,有意思的是return那一行,其中的意思在上面做了解答,查询是否有其他线程比当前线程等待获取锁花费了更多的时间,有就返回true,没有就返回false,也就是说该方法返回false,才进行addWaiter状态的更改尝试,其余和部分和非公平锁的部分一样。

ctrl点进acquire(1)是这样的:

public final void acquire(int arg) {
        if (!tryAcquire(arg) &&
            acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
            selfInterrupt();
    }
1
2
3
4
5
首先通过tryAcquire方法尝试获取锁,如果成功直接返回,否则通过acquireQueued()再次尝试获取。在acquireQueued()中会先通过addWaiter将当前线程加入到CLH队列的队尾,在CLH队列中等待。在等待过程中线程处于休眠状态,直到成功获取锁才会返回。
--------------------- 
作者:Lovnx 
来源:CSDN 
原文:https://blog.csdn.net/rickiyeat/article/details/78307739 
版权声明:本文为博主原创文章,转载请附上博文链接!

ReentrantLock与公平锁、非公平锁实现相关推荐

  1. java队列加锁_java并发-----浅析ReentrantLock加锁,解锁过程,公平锁非公平锁,AQS入门,CLH同步队列...

    前言 为什么需要去了解AQS,AQS,AbstractQueuedSynchronizer,即队列同步器.它是构建锁或者其他同步组件的基础框架(如ReentrantLock.ReentrantRead ...

  2. 公平锁非公平锁的实际使用_面经手册 · 第16篇《码农会锁,ReentrantLock之公平锁讲解和实现》...

    作者:小傅哥 博客:https://bugstack.cn 专题:面经手册 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言 Java学多少才能找到工作? 最近经常有小伙伴问我,以为我的经验来看 ...

  3. 公平锁和非公平锁-ReentrantLock是如何实现公平、非公平的

    转载:https://www.jianshu.com/p/5104cd94dbe0 1.什么是公平锁与非公平锁 公平锁:公平锁就是保障了多线程下各线程获取锁的顺序,先到的线程优先获取锁. 非公平锁:非 ...

  4. Java锁详解:“独享锁/共享锁+公平锁/非公平锁+乐观锁/悲观锁+线程锁”

    在Java并发场景中,会涉及到各种各样的锁如公平锁,乐观锁,悲观锁等等,这篇文章介绍各种锁的分类: 公平锁/非公平锁 可重入锁 独享锁/共享锁 乐观锁/悲观锁 分段锁 自旋锁 线程锁 乐观锁 VS 悲 ...

  5. Java锁之公平和非公平锁

    Java锁之公平和非公平锁 目录 公平锁和非公平锁概念 公平锁和非公平锁区别 ReentrantLock和synchronized是公平锁还是非公平锁? 1. 公平锁和非公平锁概念 公平锁:是指多个线 ...

  6. 最全Java锁详解:独享锁/共享锁+公平锁/非公平锁+乐观锁/悲观锁

    在Java并发场景中,会涉及到各种各样的锁,比如:高并发编程系列:4种常用Java线程锁的特点,性能比较.使用场景,这些锁有对应的种类:公平锁,乐观锁,悲观锁等等,这篇文章来详细介绍各种锁的分类: 公 ...

  7. 公平锁非公平锁和可重入锁(递归锁)

    公平锁非公平锁 公平锁―是指多个线程按照申请锁的顺序来获取锁,类似排队打饭,先来后到. 非公平锁是指多个线程获取锁的顺序并不是按照申请锁的顺序,有可能后申请的线程比先申请的线程优先获取锁,在高并发的情 ...

  8. 云阶月地,关锁千重(一.公平和非公平)

    看到文章的标题是不是很诧异,一个搞技术的为什么要搞这么文艺的话题呢?标题说关锁千重,是不是很形象,我们在开发中的锁不也是多种多样么? Lock 既然之前说了锁千重,那锁到底有多少种,他们的分类又是怎么 ...

  9. 可重入锁/不可重入锁,公平锁/非公平锁,乐观锁/悲观锁,独享锁/共享锁,偏向锁/轻量级锁/重量级锁,分段锁,自旋锁

    在并发编程中,会涉及到各种各样的锁,这篇文章主要介绍各种锁的分类以及作用. 介绍的内容如下: 可重入锁/不可重入锁 公平锁/非公平锁 乐观锁/悲观锁 独享锁/共享锁 偏向锁/轻量级锁/重量级锁 分段锁 ...

  10. 第二季:5公平锁/非公平锁/可重入锁/递归锁/自旋锁谈谈你的理解?请手写一个自旋锁【Java面试题】

    第二季:5值传递和引用传递[Java面试题] 前言 推荐 值传递 说明 题目 24 TransferValue醒脑小练习 第二季:5公平锁/非公平锁/可重入锁/递归锁/自旋锁谈谈你的理解?请手写一个自 ...

最新文章

  1. 基于FPGA系统合成两条视频流实现3D视频效果
  2. JavaScript最全编码规范
  3. 文件管理器_苹果超强文件管理器,秒变安卓?
  4. trie树- 转自 维基百科
  5. html改变下拉框的大小,调整屏幕大小时,HTML导航栏下拉框内容无法正确调整大小...
  6. 数学的威力:一个方程提升中国卫星图像质量30%
  7. 在VS2010中F5调试Silverlight程序时,提示“无法启动调试,找不到Microsoft Internet Explorer”...
  8. easypoi 如何合并相同的列_easy_poi合并行以及样式调整
  9. Snabbdom(虚拟dom-6-createElm函数)
  10. 游戏脚本在移动游戏设计中的作用_游戏设计中道具设计分步解说
  11. 三条中线分的六个三角形_解读三角形中的三边关系和三条线段的应用
  12. java.lang.UnsatisfiedLinkError: D:\Tomcat\apache-tomcat-7.0.67\bin\tcnative-1.dll:
  13. springboot自动配置原理_今日份学习之Spring Boot自动配置实现原理
  14. 牛b硬件信息修改大师_比X大师更靠谱?一款真正良心的硬件检测工具
  15. .net 开发工程师 面试题
  16. Linux查看最近开关机记录
  17. 微信相关开发问题收集
  18. 在移动网络上创建更稳定的连接
  19. 关闭公司电脑的强制屏幕保护
  20. [思维模式-13]:《复盘》-1- “知”篇 - 认识复盘

热门文章

  1. /bin/bash^M: bad interpreter: 没有那个文件或目录--转载
  2. spring mvc DispatcherServlet详解之前传---FrameworkServlet
  3. jboss7 加载module过程
  4. Using SSL Certificates with HAProxy--reference
  5. redis java对象操作
  6. 【风险管理】(第一篇)风险管理核心指标
  7. 【风险管理】金融产品一站式解决方案
  8. 【Python】Pycharm
  9. 评分卡开发方法论scorecard
  10. 点一万个赞:商汤SiamRPN目标跟踪最强算法开源