In addition to Weibo, there is also WeChat
Please pay attention
WeChat public account
Shulou
2025-01-18 Update From: SLTechnology News&Howtos shulou NAV: SLTechnology News&Howtos > Internet Technology >
Share
Shulou(Shulou.com)06/01 Report--
Most people do not understand the knowledge points of this article "what are the 15 locks in Java", so the editor summarizes the following content, detailed content, clear steps, and has a certain reference value. I hope you can get something after reading this article. Let's take a look at this article "what are the 15 locks in Java?"
Fair lock / unfair lock
Fair lock
Fair lock means that multiple threads acquire the lock in the order in which the lock is applied for.
Unfair lock
Unfair lock means that multiple threads do not acquire locks in the order in which they apply for locks, and it is possible that the thread that applies later takes precedence over the thread that applies first. It is possible that it can cause priority reversal or hunger.
For Java ReentrantLock, the constructor specifies whether the lock is a fair lock, and the default is an unfair lock. The advantage of unfair locks is that the throughput is larger than fair locks. For Synchronized, it is also an unfair lock. Unlike ReentrantLock, which implements thread scheduling through AQS, there is no way to make it a fair lock.
Reentrant lock / non-reentrant lock
Reentrant lock
In a broad sense, a reentrant lock refers to a repeatable and recursive lock. After the lock is used in the outer layer, it can still be used in the inner layer, and there is no deadlock (provided it is the same object or class). Such a lock is called a reentrant lock. Both ReentrantLock and synchronized are reentrant locks
Synchronized void setA () throws Exception {Thread.sleep (1000); setB ();} synchronized void setB () throws Exception {Thread.sleep (1000);}
The above code is a feature of a reentrant lock, and if it were not for the reentrant lock, the setB might not be executed by the current thread, which could cause a deadlock.
Non-reentrant lock
Non-reentrant lock, contrary to reentrant lock, cannot be called recursively, and deadlock occurs when recursive invocation occurs. See a classic explanation that uses a spin lock to simulate a non-reentrant lock, as follows
Import java.util.concurrent.atomic.AtomicReference;public class UnreentrantLock {private AtomicReference owner = new AtomicReference (); public void lock () {Thread current = Thread.currentThread (); / / this sentence is a classic "spin" syntax, and for (;;) {if (! owner.compareAndSet (null, current)) {return;} public void unlock () {Thread current = Thread.currentThread (); owner.compareAndSet (current, null);}}
The code is also relatively simple, using atomic references to store threads, the same thread calls the lock () method twice, if you do not execute unlock () to release the lock, the second call to spin will produce a deadlock, the lock is not reentrant, but in fact the same thread does not have to release the lock every time to obtain the lock, such a scheduling switch is very resource-consuming.
Turn it into a reentrant lock:
Import java.util.concurrent.atomic.AtomicReference;public class UnreentrantLock {private AtomicReference owner = new AtomicReference (); private int state = 0; public void lock () {Thread current = Thread.currentThread (); if (current = = owner.get ()) {state++; return;} / / this sentence is a classic "spin" syntax. For (;;) {if (! owner.compareAndSet (null, current)) {return;} public void unlock () {Thread current = Thread.currentThread () If (current = = owner.get ()) {if (state! = 0) {state--;} else {owner.compareAndSet (current, null);}}
Before performing each operation, determine whether the current lock holder is the current object, using state count, do not have to release the lock each time.
Implementation of reentrant lock in ReentrantLock
Here's a look at the lock acquisition method for unfair locks:
Final boolean nonfairTryAcquire (int acquires) {final Thread current = Thread.currentThread (); int c = getState (); if (c = = 0) {if (compareAndSetState (0, acquires)) {setExclusiveOwnerThread (current); return true }} / / this is where 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;} 在AQS中维护了一个private volatile int state来计数重入次数,避免了频繁的持有释放操作,这样既提升了效率,又避免了死锁。 独享锁 / 共享锁 独享锁和共享锁在你去读C.U.T包下的ReeReentrantLock和ReentrantReadWriteLock你就会发现,它俩一个是独享一个是共享锁。 独享锁 :该锁每一次只能被一个线程所持有。 共享锁 :该锁可被多个线程共有,典型的就是ReentrantReadWriteLock里的读锁,它的读锁是可以被共享的,但是它的写锁确每次只能被独占。 另外读锁的共享可保证并发读是非常高效的,但是读写和写写,写读都是互斥的。 独享锁与共享锁也是通过AQS来实现的,通过实现不同的方法,来实现独享或者共享。 对于Synchronized而言,当然是独享锁。 互斥锁 / 读写锁 互斥锁 在访问共享资源之前对进行加锁操作,在访问完成之后进行解锁操作。 加锁后,任何其他试图再次加锁的线程会被阻塞,直到当前进程解锁。 如果解锁时有一个以上的线程阻塞,那么所有该锁上的线程都被编程就绪状态, 第一个变为就绪状态的线程又执行加锁操作,那么其他的线程又会进入等待。 在这种方式下,只有一个线程能够访问被互斥锁保护的资源 读写锁 读写锁既是互斥锁,又是共享锁,read模式是共享,write是互斥(排它锁)的。 读写锁有三种状态 :读加锁状态、写加锁状态和不加锁状态 读写锁在Java中的具体实现就是 ReadWriteLock 一次只有一个线程可以占有写模式的读写锁,但是多个线程可以同时占有读模式的读写锁。 只有一个线程可以占有写状态的锁,但可以有多个线程同时占有读状态锁,这也是它可以实现高并发的原因。当其处于写状态锁下,任何想要尝试获得锁的线程都会被阻塞,直到写状态锁被释放;如果是处于读状态锁下,允许其它线程获得它的读状态锁,但是不允许获得它的写状态锁,直到所有线程的读状态锁被释放;为了避免想要尝试写操作的线程一直得不到写状态锁,当读写锁感知到有线程想要获得写状态锁时,便会阻塞其后所有想要获得读状态锁的线程。所以读写锁非常适合资源的读操作远多于写操作的情况。 乐观锁 / 悲观锁 悲观锁 总是假设最坏的情况,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会阻塞直到它拿到锁( 共享资源每次只给一个线程使用,其它线程阻塞,用完后再把资源转让给其它线程 )。传统的关系型数据库里边就用到了很多这种锁机制,比如行锁,表锁等,读锁,写锁等,都是在做操作之前先上锁。Java中synchronized和ReentrantLock等独占锁就是悲观锁思想的实现。 乐观锁 总是假设最好的情况,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,可以使用版本号机制和CAS算法实现。乐 观锁适用于多读的应用类型,这样可以提高吞吐量 ,像数据库提供的类似于write_condition机制,其实都是提供的乐观锁。在Java中java.util.concurrent.atomic包下面的 原子变量类就是使用了乐观锁的一种实现方式CAS实现的 。 分段锁 分段锁其实是一种锁的设计,并不是具体的一种锁,对于ConcurrentHashMap而言,其并发的实现就是通过分段锁的形式来实现高效的并发操作。 并发容器类的加锁机制是基于粒度更小的分段锁,分段锁也是提升多并发程序性能的重要手段之一。 在并发程序中,串行操作是会降低可伸缩性,并且上下文切换也会减低性能。在锁上发生竞争时将通水导致这两种问题,使用独占锁时保护受限资源的时候,基本上是采用串行方式--每次只能有一个线程能访问它。所以对于可伸缩性来说最大的威胁就是独占锁。 我们一般有三种方式降低锁的竞争程度 : 1、减少锁的持有时间 2、降低锁的请求频率 3、使用带有协调机制的独占锁,这些机制允许更高的并发性。 在某些情况下我们可以将锁分解技术进一步扩展为一组独立对象上的锁进行分解,这成为分段锁。 其实说的简单一点就是 : 容器里有多把锁,每一把锁用于锁容器其中一部分数据,那么当多线程访问容器里不同数据段的数据时,线程间就不会存在锁竞争,从而可以有效的提高并发访问效率,这就是ConcurrentHashMap所使用的锁分段技术,首先将数据分成一段一段的存储,然后给每一段数据配一把锁,当一个线程占用锁访问其中一个段数据的时候,其他段的数据也能被其他线程访问。 比如:在ConcurrentHashMap中使用了一个包含16个锁的数组,每个锁保护所有散列桶的1/16,其中第N个散列桶由第(N mod 16)个锁来保护。假设使用合理的散列算法使关键字能够均匀的分部,那么这大约能使对锁的请求减少到越来的1/16。也正是这项技术使得ConcurrentHashMap支持多达16个并发的写入线程。 偏向锁 / 轻量级锁 / 重量级锁 锁的状态 : 无锁状态 偏向锁状态 轻量级锁状态 重量级锁状态 锁的状态是通过对象监视器在对象头中的字段来表明的。 四种状态会随着竞争的情况逐渐升级,而且是不可逆的过程,即不可降级。 这四种状态都不是Java语言中的锁 ,而是Jvm为了提高锁的获取与释放效率而做的优化( 使用synchronized时 )。 偏向锁 偏向锁是指一段同步代码一直被一个线程所访问,那么该线程会自动获取锁。降低获取锁的代价。 轻量级 轻量级锁是指当锁是偏向锁的时候,被另一个线程所访问,偏向锁就会升级为轻量级锁,其他线程会通过自旋的形式尝试获取锁,不会阻塞,提高性能。 重量级锁 重量级锁是指当锁为轻量级锁的时候,另一个线程虽然是自旋,但自旋不会一直持续下去,当自旋一定次数的时候,还没有获取到锁,就会进入阻塞,该锁膨胀为重量级锁。重量级锁会让其他申请的线程进入阻塞,性能降低。 自旋锁 我们知道CAS算法是乐观锁的一种实现方式,CAS算法中又涉及到自旋锁,所以这里给大家讲一下什么是自旋锁。 简单回顾一下CAS算法 CAS是英文单词Compare and Swap(比较并交换),是一种有名的无锁算法。无锁编程,即不使用锁的情况下实现多线程之间的变量同步,也就是在没有线程被阻塞的情况下实现变量的同步,所以也叫非阻塞同步(Non-blocking Synchronization)。CAS算法涉及到三个操作数 需要读写的内存值 V 进行比较的值 A 拟写入的新值 B 更新一个变量的时候,只有当变量的预期值A和内存地址V当中的实际值相同时,才会将内存地址V对应的值修改为B,否则不会执行任何操作。一般情况下是一个自旋操作,即不断的重试。 什么是自旋锁? 自旋锁(spinlock):是指当一个线程在获取锁的时候,如果锁已经被其它线程获取,那么该线程将循环等待,然后不断的判断锁是否能够被成功获取,直到获取到锁才会退出循环 。 它是为实现保护共享资源而提出一种锁机制。其实,自旋锁与互斥锁比较类似,它们都是为了解决对某项资源的互斥使用。 无论是互斥锁,还是自旋锁,在任何时刻,最多只能有一个保持者,也就说,在任何时刻最多只能有一个执行单元获得锁 。但是两者在调度机制上略有不同。对于互斥锁,如果资源已经被占用,资源申请者只能进入睡眠状态。但是自旋锁不会引起调用者睡眠,如果自旋锁已经被别的执行单元保持,调用者就一直循环在那里看是否该自旋锁的保持者已经释放了锁,"自旋"一词就是因此而得名。 Java如何实现自旋锁? 下面是个简单的例子: public class SpinLock { private AtomicReference cas = new AtomicReference(); public void lock() { Thread current = Thread.currentThread(); // 利用CAS while (!cas.compareAndSet(null, current)) { // DO nothing } } public void unlock() { Thread current = Thread.currentThread(); cas.compareAndSet(current, null); }} lock()方法利用的CAS,当第一个线程A获取锁的时候,能够成功获取到,不会进入while循环,如果此时线程A没有释放锁,另一个线程B又来获取锁,此时由于不满足CAS,所以就会进入while循环,不断判断是否满足CAS,直到A线程调用unlock方法释放了该锁。 自旋锁存在的问题 1、如果某个线程持有锁的时间过长,就会导致其它等待获取锁的线程进入循环等待,消耗CPU。使用不当会造成CPU使用率极高。 2、上面Java实现的自旋锁不是公平的,即无法满足等待时间最长的线程优先获取锁。不公平的锁就会存在"线程饥饿"问题。 自旋锁的优点 1、自旋锁不会使线程状态发生切换,一直处于用户态,即线程一直都是active的;不会使线程进入阻塞状态,减少了不必要的上下文切换,执行速度快 2、非自旋锁在获取不到锁的时候会进入阻塞状态,从而进入内核态,当获取到锁的时候需要从内核态恢复,需要线程上下文切换。 (线程被阻塞后便进入内核(Linux)调度状态,这个会导致系统在用户态与内核态之间来回切换,严重影响锁的性能) 可重入的自旋锁和不可重入的自旋锁 文章开始的时候的那段代码,仔细分析一下就可以看出,它是不支持重入的,即当一个线程第一次已经获取到了该锁,在锁释放之前又一次重新获取该锁,第二次就不能成功获取到。由于不满足CAS,所以第二次获取会进入while循环等待,而如果是可重入锁,第二次也是应该能够成功获取到的。 而且,即使第二次能够成功获取,那么当第一次释放锁的时候,第二次获取到的锁也会被释放,而这是不合理的。 为了实现可重入锁,我们需要引入一个计数器,用来记录获取锁的线程数。 public class ReentrantSpinLock { private AtomicReference cas = new AtomicReference(); private int count; public void lock() { Thread current = Thread.currentThread(); if (current == cas.get()) { // 如果当前线程已经获取到了锁,线程数增加一,然后返回 count++; return; } // 如果没获取到锁,则通过CAS自旋 while (!cas.compareAndSet(null, current)) { // DO nothing } } public void unlock() { Thread cur = Thread.currentThread(); if (cur == cas.get()) { if (count >0) {/ / if it is greater than 0, it means that the current thread has acquired the lock several times. Releasing the lock simulates count--;} else {/ / if count==0 is reduced by 1 by count, the lock can be released, which ensures that the number of times the lock is acquired is consistent with the number of times the lock is released. Cas.compareAndSet (cur, null);}}
Spin lock and mutex lock
Both spin lock and mutex lock are mechanisms to protect resource sharing.
Whether it is a spin lock or a mutex lock, there can be at most one holder at any one time.
The thread that acquires the mutex will go to sleep if the lock is already occupied; the thread that acquires the spin lock will not sleep, but will loop and wait for the lock to be released.
Here I recommend my Java back-end technology circle: 867857579, there are (distributed architecture, high scalability, high performance, high concurrency, performance optimization, Spring boot, Redis, ActiveMQ, and other learning resources) to every Java partner free of charge, whether you change your profession or want to improve your ability at work, welcome to join the group for in-depth exchange and learning!
Spin lock summary
Spin lock: when a thread acquires a lock, if the lock is held by another thread, the current thread will loop and wait until the lock is acquired.
During the spin lock wait, the state of the thread does not change, and the thread is always in user mode and active.
A spin lock that holds the lock for too long will cause other threads waiting to acquire the lock to run out of CPU.
The spin lock itself cannot guarantee fairness, nor can it guarantee reentrability.
Based on the spin lock, the lock with fairness and reentrant property can be realized.
The above is about the content of this article on "what are the 15 locks in Java". I believe we all have a certain understanding. I hope the content shared by the editor will be helpful to you. If you want to know more about the relevant knowledge, please follow the industry information channel.
Welcome to subscribe "Shulou Technology Information " to get latest news, interesting things and hot topics in the IT industry, and controls the hottest and latest Internet news, technology news and IT industry trends.
Views: 0
*The comments in the above article only represent the author's personal views and do not represent the views and positions of this website. If you have more insights, please feel free to contribute and share.
Continue with the installation of the previous hadoop.First, install zookooper1. Decompress zookoope
"Every 5-10 years, there's a rare product, a really special, very unusual product that's the most un
© 2024 shulou.com SLNews company. All rights reserved.