前言

Java语言中有许多原生线程安全的数据结构,比如ArrayBlockingQueue、CopyOnWriteArrayList、LinkedBlockingQueue,它们线程安全的实现方式并非通过synchronized关键字,而是通过java.util.concurrent.locks.ReentrantLock来实现。
 

锁的底层实现

无论什么语言在操作系统层面锁的操作都会变成系统调用(System Call),以 Linux 为例,就是 futex 函数,可以把它理解为两个函数:futex_wait(s),对变量 s 加锁;futex_wake(s)释放 s 上的锁,唤醒其他线程。
在ReentrantLock中很明显可以看到其中同步包括两种,分别是公平的FairSync和非公平的NonfairSync。
公平锁的作用就是严格按照线程启动的顺序来执行的,不允许其他线程插队执行的;而非公平锁是允许插队的。
默认情况下ReentrantLock是通过非公平锁来进行同步的,包括synchronized关键字都是如此,因为这样性能会更好。
因为从线程进入了RUNNABLE状态,可以执行开始,到实际线程执行是要比较久的时间的。
而且,在一个锁释放之后,其他的线程会需要重新来获取锁。其中经历了持有锁的线程释放锁,其他线程从挂起恢复到RUNNABLE状态,其他线程请求锁,获得锁,线程执行,这一系列步骤。如果这个时候,存在一个线程直接请求锁,可能就避开挂起到恢复RUNNABLE状态的这段消耗,所以性能更优化。
 
    /**
* Creates an instance of {@code ReentrantLock}.
* This is equivalent to using {@code ReentrantLock(false)}.
*/
public ReentrantLock() {
sync = new NonfairSync();
}
默认状态,使用的ReentrantLock()就是非公平锁。再参考如下代码,我们知道ReentrantLock的获取锁的操作是通过装饰模式代理给sync的。
 
    /**
* Acquires the lock.
*
* <p>Acquires the lock if it is not held by another thread and returns
* immediately, setting the lock hold count to one.
*
* <p>If the current thread already holds the lock then the hold
* count is incremented by one and the method returns immediately.
*
* <p>If the lock is held by another thread then the
* current thread becomes disabled for thread scheduling
* purposes and lies dormant until the lock has been acquired,
* at which time the lock hold count is set to one.
*/
public void lock() {
sync.lock();
}
下面参考一下FairSync和NonfairSync对lock方法的实现:
 
    /**
* Sync object for non-fair locks
*/
static final class NonfairSync extends Sync {
/**
* Performs lock. Try immediate barge, backing up to normal
* acquire on failure.
*/
final void lock() {
if (compareAndSetState(0, 1))
setExclusiveOwnerThread(Thread.currentThread());
else
acquire(1);
}
} /**
* Sync object for fair locks
*/
static final class FairSync extends Sync {
final void lock() {
acquire(1);
}
}
当使用非公平锁的时候,会立刻尝试配置状态,成功了就会插队执行,失败了就会和公平锁的机制一样,调用acquire()方法,以排他的方式来获取锁,成功了立刻返回,否则将线程加入队列,知道成功调用为止。
 

总结

上锁的过程本身也是有时间开销的,如果操作资源的时间比上锁的时间还短建议使用非公平锁可以提高系统的吞吐率;否则就老老实实的用公平锁。
 

最后

欢迎大家关注我的公种浩【程序员追风】,2019年多家公司java面试题整理了1000多道400多页pdf文档,文章都会在里面更新,整理的资料也会放在里面。
喜欢文章记得关注我点个赞哟,感谢支持!

 

最新文章

  1. NodeJs之pm2
  2. C语言中的运算符
  3. libgdx 常见问题
  4. 【Log4j2 配置详解】log4j2的资源文件具体怎么配置
  5. JQuery:JQuery添加元素
  6. JS中的this用法详解
  7. Windows和Linux下查看Apache、MySQL、PHP版本
  8. Map集合的四种遍历
  9. json &lt;---&gt;List集合,实体类 之间的相互转换
  10. vmstat
  11. AsyncTask兼容性
  12. javascript排序、功能代码总结[长期更新]
  13. RE模块错误已解决.
  14. 网络I/O 工作机制
  15. Java问题解决:&quot;错误:编码GBK 的不可映射字符&quot;
  16. 百度地图API---JS开发
  17. VS unable to update auto-refresh path。。。。
  18. [Java初探08]__简单学习Java类和对象
  19. UVa 1252 - Twenty Questions(状压DP)
  20. IOS项目之弹出动画终结篇

热门文章

  1. Activity生命周期理解
  2. c# 发送邮箱,企业邮箱测试成功
  3. TypeScript SDK 和 REST API
  4. 20191017-7 alpha week 2/2 Scrum立会报告+燃尽图 06
  5. Nginx-(四)基本模块2
  6. 12、pytest -- 缓存:记录执行的状态
  7. myql数据库,sql横排转竖排以及竖排转横排,oracle的over函数的使用
  8. Nginx配置详细解析(全)
  9. CSRF的原理与防御 | 你想不想来一次CSRF攻击?
  10. python推导式pythonic必备【华为云技术分享】