Java ReEntrantLock (Java代码实战-001)详解编程语言

Lock类也可以实现线程同步,而Lock获得锁需要执行lock方法,释放锁需要执行unLock方法

Lock类可以创建Condition对象,Condition对象用来使线程等待和唤醒线程,需要注意的是Condition对象的唤醒的是用同一个Condition执行await方法的线程,所以也就可以实现唤醒指定类的线程

Lock类分公平锁和不公平锁,公平锁是按照加锁顺序来的,非公平锁是不按顺序的,也就是说先执行lock方法的锁不一定先获得锁 Lock类有读锁和写锁,读读共享,写写互斥,读写互斥

 

import java.sql.Date; 
import java.util.concurrent.ExecutorService; 
import java.util.concurrent.Executors; 
import java.util.concurrent.TimeUnit; 
import java.util.concurrent.locks.Condition; 
import java.util.concurrent.locks.ReentrantLock; 
 
/** 
 * ReEntrantLockTest 
 */ 
public class ReEntrantLockTest { 
    private static final int THREAD_COUNT = 4; 
    private static final int WAIT_TIMES = 5; 
 
    public static void main(String[] args) { 
        // 默认是非公平锁 
        final ReentrantLock lock = new ReentrantLock(); 
 
        final Condition con = lock.newCondition(); 
        final Runnable add = new Runnable() { 
            @Override 
            public void run() { 
                System.out.println("Time: " + new java.util.Date() + " Pre " + lock); 
                // 获取到锁的话立即返回,否则一直处于等待状态 
                lock.lock(); 
                try { 
                    // 使当前线程处于等待状态,直到被信号通知或中断,此线程处于等待状态,其它线程也可以获取锁 
                    con.await(WAIT_TIMES, TimeUnit.SECONDS); 
 
                    // seleep状态时则不会释放锁 
                    Thread.sleep(5000); 
                } catch (InterruptedException e) { 
                    e.printStackTrace(); 
                } finally { 
                    System.out.println("Time:" + new java.util.Date() + " Post " + lock); 
                    lock.unlock(); 
                } 
            } 
        }; 
 
        final ExecutorService exec = Executors.newFixedThreadPool(THREAD_COUNT); 
        for (int i = 0; i < THREAD_COUNT; i++) { 
            exec.submit(add); 
        } 
        exec.shutdown(); 
    } 
}

 

ReenTrantLock可重入锁(和synchronized的区别)总结

可重入性:

从名字上理解,ReenTrantLock的字面意思就是再进入的锁,其实synchronized关键字所使用的锁也是可重入的,两者关于这个的区别不大。两者都是同一个线程没进入一次,锁的计数器都自增1,所以要等到锁的计数器下降为0时才能释放锁。

锁的实现:

Synchronized是依赖于JVM实现的,而ReenTrantLock是JDK实现的,有什么区别,说白了就类似于操作系统来控制实现和用户自己敲代码实现的区别。前者的实现是比较难见到的,后者有直接的源码可供阅读。

性能的区别:

在Synchronized优化以前,synchronized的性能是比ReenTrantLock差很多的,但是自从Synchronized引入了偏向锁,轻量级锁(自旋锁)后,两者的性能就差不多了,在两种方法都可用的情况下,官方甚至建议使用synchronized,其实synchronized的优化我感觉就借鉴了ReenTrantLock中的CAS技术。都是试图在用户态就把加锁问题解决,避免进入内核态的线程阻塞。

功能区别:

便利性:很明显Synchronized的使用比较方便简洁,并且由编译器去保证锁的加锁和释放,而ReenTrantLock需要手工声明来加锁和释放锁,为了避免忘记手工释放锁造成死锁,所以最好在finally中声明释放锁。

锁的细粒度和灵活度:很明显ReenTrantLock优于Synchronized

ReenTrantLock独有的能力:

1.      ReenTrantLock可以指定是公平锁还是非公平锁而synchronized只能是非公平锁。所谓的公平锁就是先等待的线程先获得锁。

2.      ReenTrantLock提供了一个Condition(条件)类,用来实现分组唤醒需要唤醒的线程们,而不是像synchronized要么随机唤醒一个线程要么唤醒全部线程。

3.      ReenTrantLock提供了一种能够中断等待锁的线程的机制,通过lock.lockInterruptibly()来实现这个机制。

ReenTrantLock实现的原理:

在网上看到相关的源码分析,本来这块应该是本文的核心,但是感觉比较复杂就不一一详解了,简单来说,ReenTrantLock的实现是一种自旋锁,通过循环调用CAS操作来实现加锁。它的性能比较好也是因为避免了使线程进入内核态的阻塞状态。想尽办法避免线程进入内核的阻塞状态是我们去分析和理解锁设计的关键钥匙。

什么情况下使用ReenTrantLock:

答案是,如果你需要实现ReenTrantLock的三个独有功能时。

 

原创文章,作者:ItWorker,如若转载,请注明出处:https://blog.ytso.com/tech/pnotes/15932.html

(0)
上一篇 2021年7月19日 18:51
下一篇 2021年7月19日 18:51

相关推荐

发表回复

登录后才能评论