上好的锁机制

什么是乐观锁

乐观锁是一种乐观思想,即认为读多写少,遇到并发写的可能性低,每次去拿 数据 的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个 数据 ,采取 在写时先读出当前版本号,然后加锁操作(比较跟上一次的版本号,如果一样则更新),如果失败则要 重复读-比较-写的操作。java 中的乐观锁基本都是通过 CAS操作实现的,CAS是一种更新的原子操作, 比较当前值跟传入 值是否一样,一样则更新,否则失败。

2、什么是悲观锁

悲观锁是就是悲观思想,即认为写多,遇到并发写的可能性高,每次去拿 数据 的时候都认为别人会 修改,所以每次在读写 数据 的时候都会上锁,这样别人想读写这个 数据 就会 block 直到拿到锁。java中 的悲观锁就是Synchronized,AQS框架下的锁则是先尝试cas乐观锁去获取锁,获取不到,才会转换为悲 观锁,如RetreenLock。

3、什么是自旋锁

自旋锁原理非常简单, 如果持有锁的线程能在很短时间内释放锁资源,那么那些等待竞争锁的线 程就不需要做内核态和用户态之间的切换进入阻塞挂起状态,它们只需要等一等(自旋),等持有锁的 线程释放锁后即可立即获取锁,这样就避免用户线程和内核的切换的消耗。线程自旋是需要消耗 cup 的,说白了就是让 cup 在做无用功,如果一直获取不到锁,那线程也不 能一直占用 cup 自旋做无用功,所以需要设定一个自旋等待的最大时间。如果持有锁的线程执行的时间超过自旋等待的最大时间扔没有释放锁,就会导致其它争用锁的线程 在最大等待时间内还是获取不到锁,这时争用线程会停止自旋进入阻塞状态。

4、自旋锁的优缺点

优点

自旋锁尽可能的减少线程的阻塞,这对于锁的竞争不激烈,且占用锁时间非常短的代码块来说性能 能大幅度的提升,因为自旋的消耗会小于线程阻塞挂起再唤醒的操作的消耗,这些操作会导致线程发生 两次上下文切换!

缺点

但是如果锁的竞争激烈,或者持有锁的线程需要长时间占用锁执行同步块,这时候就不适合使用自 旋锁了,因为自旋锁在获取锁前一直都是占用 cpu 做无用功,占着 XX 不 XX,同时有大量线程在竞争一 个锁,会导致获取锁的时间很长,线程自旋的消耗大于线程阻塞挂起操作的消耗,其它需要 cup 的线程 又不能获取到cpu,造成 cpu 的浪费。所以这种情况下我们要关闭自旋锁;自旋锁时间阈值(1.6 引入了适应性自旋锁)自旋锁的目的是为了占着 CPU 的资源不释放,等到获取到锁立即进行处理。但是如何去选择自旋 的执行时间呢?如果自旋执行时间太长,会有大量的线程处于自旋状态占用 CPU 资源,进而会影响整 体系统的性能。因此自旋的周期选的额外重要!JVM对于自旋周期的选择, jdk1.5 这个限度是一定的写死的, 在 1.6 引入了适应性自旋锁,适应 性自旋锁意味着自旋的时间不在是固定的了,而是由前一次在同一个锁上的自旋时间以及锁的拥有者的 状态来决定,基本认为一个线程上下文切换的时间是最佳的一个时间,同时JVM 还针对当前 CPU 的负 荷情况做了较多的优化, 如果平均负载小于 CPUs 则一直自旋, 如果有超过(CPUs/2)个线程正在自 旋,则后来线程直接阻塞, 如果正在自旋的线程发现 Owner 发生了变化则延迟自旋时间(自旋计数) 或进入阻塞, 如果 CPU 处于节电模式则停止自旋, 自旋时间的最坏情况是 CPU的存储延迟(CPU A 存储了一个 数据 ,到 CPU B 得知这个 数据 直接的时间差) , 自旋时会适当放弃线程优先级之间的差 异。

自旋锁的开启

JDK1.6 中-XX:+UseSpinning 开启;

-XX:PreBlockSpin=10 为自旋次数;

JDK1.7 后,去掉此参数,由 jvm 控制;

5、Synchronized 同步锁

synchronized 它可以把任意一个非 NULL 的对象当作锁。 他属于独占式的悲观锁,同时属于可重 入锁 。

synchronized 作用范围

  1. 作用于方法时,锁住的是对象的实例(this);

  2. 当作用于静态方法时,锁住的是Class实例,又因为Class的相关 数据 存储在永久带PermGen(jdk1.8 则是 metaspace), 永久带是全局共享的,因此静态方法锁相当于类的一个全 局锁,会锁所有调用该方法的线程;

  3. synchronized 作用于一个对象实例时,锁住的是所有以该对象为锁的代码块。 它有多个队列, 当多个线程一起访问某个对象监视器的时候,对象监视器会将这些线程存储在不同的容器中。

Synchronized 核心组件

  1. Wait Set:哪些调用 wait 方法被阻塞的线程被放置在这里;

  2. Contention List: 竞争队列,所有请求锁的线程首先被放在这个竞争队列中;

  3. Entry List: Contention List 中那些有资格成为候选资源的线程被移动到 Entry List 中;

  4. OnDeck:任意时刻, 最多只有一个线程正在竞争锁资源,该线程被成为 OnDeck;

  5. Owner:当前已经获取到所资源的线程被称为 Owner;

  6. !Owner:当前释放锁的线程。

Synchronized 实现

  1. JVM每次从队列的尾部取出一个 数据 用于锁竞争候选者(OnDeck),但是并发情况下,

ContentionList 会被大量的并发线程进行 CAS访问,为了降低对尾部元素的竞争, JVM会将一部 分线

程移动到 EntryList 中作为候选竞争线程。

  1. Owner 线程会在 unlock 时,将 ContentionList 中的部分线程迁移到 EntryList 中,并指定

EntryList 中的某个线程为 OnDeck 线程(一般是最先进去的那个线程)。

  1. Owner 线程并不直接把锁传递给 OnDeck 线程,而是把锁竞争的权利交给 OnDeck,OnDeck需要重

新竞争锁。这样虽然牺牲了一些公平性,但是能极大的提升系统的吞吐量,在JVM中,也把 这种选择

行为称之为“竞争切换”。

  1. OnDeck 线程获取到锁资源后会变为 Owner 线程,而没有得到锁资源的仍然停留在 EntryList中。 如果Owner 线程被 wait 方法阻塞,则转移到 WaitSet 队列中, 直到某个时刻通过 notify或者 notifyAll 唤醒 ,会重新进去 EntryList中。

  2. 处于 ContentionList

、 EntryList、 WaitSet 中的线程都处于阻塞状态,该阻塞是由操作系统来完 成的(Linux 内核下采用 pthread_mutex_lock 内核函数 实 现的)。

  1. Synchronized 是非公平锁。 Synchronized 在线程进入 ContentionList 时, 等待的线程会先尝试 自旋

获取锁,如果获取不到就进入 ContentionList,这明显对于已经进入队列的线程是不公平 的,还有一个

不公平的事情就是自旋获取锁的线程还可能直接抢占 OnDeck 线程的锁资源。

  1. 每个对象都有个 monitor 对象, 加锁就是在竞争 monitor 对象,代码块加锁是在前后分别加上

monitorenter 和 monitorexit 指令来实现的,方法加锁是通过一个标记位来判断的

  1. synchronized 是一个重量级操作,需要调用操作系统相关接口,性能是低效的,有可能给线程加 锁消

耗的时间比有用操作消耗的时间更多。

  1. Java1.6, synchronized 进行了很多的优化, 有适应自旋、锁消除、锁粗化、轻量级锁及偏向锁 等,

效率有了本质上的提高。在之后推出的 Java1.7 与 1.8 中,均对该关键字的实现机理做了优 化。引入

了偏向锁和轻量级锁。都是在对象头中有标记位,不需要经过操作系统加锁。

  1. 锁可以从偏向锁升级到轻量级锁,再升级到重量级锁。这种升级过程叫做锁膨胀;

  2. JDK1.6 中默认是开启偏向锁和轻量级锁,可以通过-XX:-UseBiasedLocking 来禁用偏向锁。

6、ReentrantLockReentantLock 继承接口

Lock 并实现了接口中定义的方法, 他是一种可重入锁, 除了能完成

synchronized 所能完成的所有工作外,还提供了诸如可响应中断锁、可轮询锁请求、定时锁等避免多 线程死锁的方法。

7、Lock 接口的主要方法

void lock(): 执行此方法时, 如果锁处于空闲状态, 当前线程将获取到锁.

相反 ,

如果锁已经被其他线 程持 有, 将禁用当前线程, 直到当前线程获取到锁.

1、boolean tryLock()

如果锁可用, 则获取锁, 并立即返回 true, 否则返回 false. 该方法和lock()的区别在于

1、tryLock()

只是"试图"获取锁, 如果锁不可用, 不会导致当前线程被禁用,当前线程仍然继续往下执 行代码.

2、lock()

是一定要获取到锁, 如果锁不可用, 就一直等待, 在未获得锁之前,当前线程并不继 续向下执行.

2、void unlock():

执行此方法时, 当前线程将释放持有的锁. 锁只能由持有者释放, 如果线程并不持有 锁,却执行该方法, 可能导致异常的发生

3、getHoldCount()

查询当前线程保持此锁的次数,也就是执行此线程执行 lock 方法的次数。

4、getQueueLength()

返回正等待获取此锁的线程估计数,比如启动 10 个线程, 1 个线程获得锁,此时返回的是 9

5、getWaitQueueLength(Condition condition)

返回等待与此锁相关的给定条件的线程估计 数。比如 10 个线程,

用同一个 condition 对象,并且此时这 10 个线程都执行了condition 对象的 await 方法,那么此时执行此方法返回 10

6、hasWaiters(Condition condition)

查询是否有线程等待与此锁有关的给定条件(condition),对于 指定contidion 对象,有多少线程执行了 condition.await 方法

7、Condition newCondition()

条件对象,获取等待通知组件。该组件和当前的锁绑定,当前线程只 有获取了锁,

才能调用该组件的 await()方法,而调用后,当前线程将缩放锁。

8、hasQueuedThread(Thread thread)

查询给定线程是否等待获取此锁

9、hasQueuedThreads()

是否有线程等待此锁

10、isFair()

该锁是否公平锁

11、sHeldByCurrentThread()

当前线程是否保持锁锁定,线程的执行 lock 方法的前后分别是 false 和true

12、isLock()

此锁是否有任意线程占用

13、lockInterruptibly()

如果当前线程未被中断,获取锁

14、tryLock()

尝试获得锁,仅在调用时锁未被线程占用,获得锁

15、tryLock(long timeout TimeUnit unit)

如果锁在给定等待时间内没有被另一个线程保持,则获取 该锁 。

8、公平锁

公平锁指的是锁的分配机制是公平的,通常先对锁提出获取请求的线程会先被分配到锁,

ReentrantLock 在构造函数中提供了是否公平锁的初始化方式来定义公平锁。40、Condition 类和 Object 类锁方法区别区别

  1. Condition 类的 awiat 方法和 Object 类的 wait 方法等效

  2. Condition 类的 signal 方法和 Object 类的 notify 方法等效

  3. Condition 类的 signalAll 方法和 Object 类的 notifyAll 方法等效

  4. ReentrantLock 类可以唤醒指定条件的线程,而 object 的唤醒是随机的

9、非公平锁

JVM按随机、就近原则分配锁的机制则称为不公平锁, ReentrantLock 在构造函数中提供了是否 公平锁

的初始化方式,默认为非公平锁。 非公平锁实际执行的效率要远远超出公平锁,除非程序有特殊 需要,否则

最常用非公平锁的分配机制。

10、tryLock 和 lock 和 lockInterruptibly 的区别

  1. tryLock 能获得锁就返回 true,不能就立即返回 false,

tryLock(long timeout,TimeUnit unit),可以增加时间限制,如果超过该时间段还没获得锁,返回 false

  1. lock 能获得锁就返回 true,不能的话一直等待获得锁

  2. lock 和 lockInterruptibly,如果两个线程分别执行这两个方法,但此时中断这两个线程,lock 不 会抛

出异常,而 lockInterruptibly 会抛出异常。

11、可重入锁(递归锁)

本文里面讲的是广义上的可重入锁,而不是单指 JAVA 下的 ReentrantLock。可 重入锁 ,也叫做 递归锁 ,指的是同一线程 外层函数获得锁之后 ,内层递归函数仍然有获取该锁的代码,但不受影响。在 JAVA 环境下 ReentrantLock 和 synchronized 都是 可重入锁。

12、ReadWriteLock 读写锁为了提高性能,

Java 提供了读写锁,在读的地方使用读锁,在写的地方使用写锁,灵活控制,如果没有写锁的情况下,读是无阻塞的,在一定程度上提高了程序的执行效率。 读写锁分为读锁和写锁,多个读锁不互斥,读锁与写锁互斥,这是由 jvm 自己控制的,你只要上好相应的锁即可。

读锁

如果你的代码只读 数据 ,可以很多人同时读,但不能同时写,那就上读锁

写锁

如果你的代码修改 数据 ,只能有一个人在写,且不能同时读取,那就上写锁。总之,读的时候上读 锁,写的时候上写锁!Java中读写锁有个接java.util.concurrent.locks.ReadWriteLock,也有具体的实现ReentrantReadWriteLock。

13、共享锁和独占锁

java 并发包提供的加锁模式分为独占锁和共享锁。

独占锁

独占锁模式下,每次只能有一个线程能持有锁, ReentrantLock 就是以独占方式实现的互斥锁。

独占锁是一种悲观保守的加锁策略,它避免了读/读冲突,如果某个只读线程获取锁,

则其他读线程都 只能等待,这种情况下就限制了不必要的并发性,因为读操作并不会影响 数据 的一致性。

共享锁

共享锁则允许多个线程同时获取锁,并发访问 共享资源,如: ReadWriteLock。

共享锁则是一种 乐观锁,它放宽了加锁策略,允许多个执行读操作的线程同时访问共享资源。

  1. AQS 的内部类 Node 定义了两个常量 SHARED和 EXCLUSIVE,他们分别标识 AQS 队列中等

待线程的锁获取模式。

  1. java 的并发包中提供了 ReadWriteLock,读-写锁。它允许一个资源可以被多个读操作访问,

或者被一个 写操作访问,但两者不能同时进行。

你可能感兴趣的:(java,大数据,jvm)