从synchronized和lock区别入手聊聊java锁机制
写这篇文章之前,我去百度了一下啥叫锁,百度百科上写道:置于可启闭的器物上,以钥匙或暗码开启。确实我们一般理解的锁就是门锁,密码锁,但是在计算机科学中,锁又是啥,说实话,这个问题我也思考了很久,也没法很好的用一两句话就让人听得明白,也不想有人看到我的文章,然后将我的结论当作答案,我觉得最好的答案还是在探索的过程中得到的,接下来我们就好好探索一番。
作为一名java程序员,最开始接触到的锁就是synchronized,书本上是这么写的,老师也是这么说的,至于为啥叫锁,可能也没多少人真的去思考过。不知道有没有同学和我一样,经历过只知道用synchronized,后来逐渐的了解ReentrantLock,读写锁,然后又了解了aqs,后来通过百度google,看一些博客(这个我要吐槽一下,在学习过程中遇到过很多文章写的有问题的,反而误导了我),后面看了看synchronized的源码,最后对比synchronized和ReentrantLock才加深了对锁的一些认知(说实话,作为一个刚毕业3年的非科班出身码农,我也不敢保证自己写的就一定对,算是学习过程中的一些感悟吧),那接下来我就按照学习顺序来逐渐展开。
先来一段简单的synchronized使用代码:
public static void main(String[] args) {
String s = new String();
synchronized (s) {
TestJni jni = new TestJni();
jni.jniHello();
}
}
上面代码做的事情很简单,如下图所示,有A B C D E多个线程同时来到synchronized包含的代码块,A先一步进来了,那么BCDE都得等,等我A执行完他们才能进来执行。

synchronized用起来确实很简单,我们也可以放在方法上,但是其本质还是锁的对象,这个我们后面分析源码一看就知道了。
随着开发时间越长,synchronized在有些复杂场景下(如需要可中断,可控制时间抢锁,需要多个等待队列分别控制,读写锁等场景的时候)无法满足我们的需求,那么就要用到Lock,下面我们先介绍一下Lock的简单使用:
Lock lock = new ReentrantLock();
lock.lock();
try {
System.out.println("线程:"+Thread.currentThread().getName()+ " 进来啦");
}finally {
lock.unlock();
}
上面是一种最简单的使用,和synchronized作用是一样的,不过加锁之后必须要解锁,且必须紧跟try - finally块解锁,使用起来稍微复杂一点,容易出错。
我们再介绍一种可中断的使用方式:
public static void main(String[] args) {
Thread thread = new Thread(() -> {
try {
lock.lockInterruptibly();
try {
testLock();
} finally {
lock.unlock();
}
} catch (Exception e){
}
});
thread.start();
thread.interrupt();
}
public static void testLock(){
condition.signalAll();
System.out.println("线程:"+Thread.currentThread().getName()+ " 进来啦");
}
这种方式呢,在拿锁被park住了,如果刚好这时候被打断了,就会响应打断退出抢锁并抛出异常,至于捕获到异常开发者怎么做,那就得根据业务来分别处理了。
而像可控制时间的其实就要稍微复杂一点,先看一下synchronized中的使用:
static TestHash s = new TestHash();
public static void main(String[] args) {
Thread thread1 = new Thread(()->{
testLock();
});
Thread thread2 = new Thread(()->{
synchronized (s) {
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
}
s.notify();
testLock();
}
});
thread1.start();
thread2.start();
} public static void testLock(){
synchronized (s) {
System.out.println("线程:"+Thread.currentThread().getName()+ " 进来啦");
try {
s.wait();
System.out.println("线程:"+Thread.currentThread().getName()+ " 叫醒啦");
} catch (InterruptedException e) {
System.out.println("抛异常啦");
}
}
}
这个例子看着要比前面几个复杂一点,首先thread1会进入testLock方法,并拿到锁,thread2等了1秒叫醒thread1(这里就是简单的wait/notify的使用),然后在拿到锁的情况下,再次进入testLock方法并拿到锁,由于没人唤醒了,会一直卡在这里(这里证明了synchronized的可重入),结果我就不贴了,感兴趣的可以拿着代码去试。
而ReentrantLock的使用也差不多,就是提前用lock去new一个Condition:
static ReentrantLock lock = new ReentrantLock();
static Condition condition = lock.newCondition();
public static void main(String[] args) {
Thread thread1 = new Thread(()->{
testWaitSingal();
},"thread1");
Thread thread2 = new Thread(()->{
lock.lock();
try {
TimeUnit.SECONDS.sleep(1);
condition.signal();
testWaitSingal();
} catch (InterruptedException e) {
}finally {
lock.unlock();
}
},"thread2");
thread1.start();
thread2.start();
} public static void testWaitSingal(){
lock.lock();
try {
System.out.println("线程:"+Thread.currentThread().getName()+ " 进来啦");
condition.await();
System.out.println("线程:"+Thread.currentThread().getName()+ " 叫醒啦");
} catch (InterruptedException e) {
System.out.println("抛异常啦");
}finally {
lock.unlock();
}
}
可以看到两种用法基本上是一致的,也就是将synchronized换成了lock,wait换成await,notify换成singal,
总结:
基本上我们平时用到的synchronized关键字的用法也就这些,但lock锁不一样,它还支持如上述的中断,更复杂的读写锁,还可以在aqs的基础上衍生出更多,如countDownLatch,cyclicBarrier等,可以支持我们做更多,但是不是lock就可以完全替代synchronized了呢,其实synchronized也有自己的优点,简单,不易出错,性能也不比lock差(有的书上写道synchronized性能比lock好,但其实就算好也不会好太多,对于我们来说,基本上可以忽略),真要说选哪个,我的建议是优先选synchronized,如果有特殊业务特殊需求synchronized无法满足,那当然是要用lock,不过,一定要记得释放锁哦。
本来打算结合reentrant和synchronized直接串起来讲的,但是确实有点多,这一篇就当作是后面的引子吧。
从synchronized和lock区别入手聊聊java锁机制的更多相关文章
- 转 : 深入解析Java锁机制
深入解析Java锁机制 https://mp.weixin.qq.com/s?__biz=MzU0OTE4MzYzMw%3D%3D&mid=2247485524&idx=1&s ...
- Java 锁机制总结
锁的种类 独享锁 VS 共享锁 独享锁:锁只能被一个线程持有(synchronized) 共享锁:锁可以被多个程序所持有(读写锁) 乐观锁 VS 悲观锁 乐观锁:每次去拿数据的时候都乐观地认为别人不会 ...
- java锁机制的面试题
java锁机制的面试题 1.ABA问题 2.CAS乐观锁 3.synchronize实现原理 4.synchronize与lock的区别 5.volatile实现原理 6.乐观锁的业务场景及实现方式 ...
- Java锁机制深入理解
Java锁机制 背景知识 指令流水线 CPU的基本工作是执行存储的指令序列,即程序.程序的执行过程实际上是不断地取出指令.分析指令.执行指令的过程. 几乎所有的冯•诺伊曼型计算机的CPU,其工 ...
- java 锁机制(synchronized 与 Lock)
在java中,解决同步问题,很多时候都会使用到synchronized和Lock,这两者都是在多线程并发时候常使用的锁机制. synchronized是java中的一个关键字,也就是说是java内置的 ...
- Java Lock Example – ReentrantLock(java锁的例子)
Welcome to Java Lock example tutorial. Usually when working with multi-threaded environment, we use ...
- 聊聊Java SPI机制
一.Java SPI机制 SPI(Service Provider Interface)是JDK内置的服务发现机制,用在不同模块间通过接口调用服务,避免对具体服务服务接口具体实现类的耦合.比如JDBC ...
- JAVA锁机制-可重入锁,可中断锁,公平锁,读写锁,自旋锁,
如果需要查看具体的synchronized和lock的实现原理,请参考:解决多线程安全问题-无非两个方法synchronized和lock 具体原理(百度) 在并发编程中,经常遇到多个线程访问同一个 ...
- 【面试专栏】JAVA锁机制
1. 悲观锁 / 乐观锁 在Java和数据库中都存在悲观锁和乐观锁的应用.Mysql锁机制中的悲观锁和乐观锁请查看: Mysql锁机制--悲观锁和乐观锁 悲观锁:在获得数据时先加锁,只到数 ...
随机推荐
- RocketMQ(七):高性能探秘之MappedFile
RocketMQ作为消息中间件,经常会被用来和其他消息中间件做比较,比对rabbitmq, kafka... 但个人觉得它一直对标的,都是kafka.因为它们面对的场景往往都是超高并发,超高性能要求的 ...
- 【实时渲染】实时3D渲染如何加速汽车线上体验应用推广
在过去,一支优秀的广告片足以让消费者对一辆汽车产生兴趣.完美的底盘线条或引擎的轰鸣声便会让潜在买家跑到经销商那里试驾.现在,广告还是和往常一样,并没有失去其特性,但86%的买家在与销售交流之前会在网上 ...
- 浅析Python闭包
1.什么是闭包 在介绍闭包概念前,我们先来看一段简短的代码 def sum_calc(*args): def wrapper(): sum = 0 for n in args: sum += n; r ...
- 记一次Apache的代码导致生产服务耗时增加
引言 二狗:二胖快醒醒,赶紧看看刚才报警邮件,你上次写的保存用户接口耗时(<二胖的参数校验坎坷之路>)大大上升,赶紧排查下原因. 二胖:好的,马上看,内心戏可十足(心里却在抱怨,大中午的搅 ...
- IDEA 远程debug
远程debug tomcat 的Catalina.sh 里面有个参数 JPDA_ADDRESS="5555",默认为5555.启动tomcat时,用 ./catalina.sh j ...
- ProxyDroid+wifi设置抓xx点评抓不到的包
ProxyDroid+wifi设置抓xx点评抓不到的包 一.ProxyDroid介绍 1.下载地址 下载并安装ProxyDroid.(确保您的设备已经ROOT)点击下载 2.界面介绍 (1) Auto ...
- kill的使用
Linux中Kill进程的N种方法 (2011-12-23 17:27:59) 转载▼ 标签: 杂谈 分类: ubuntu系统操作 常规篇: 首先,用ps查看进程,方法如下: $ ps -ef -- ...
- lambda表达式之方法引用
/** * 方法引用提供了非常有用的语法,可以直接引用已有Java类或对象(实例)的方法或构造器.<br> * 与lambda联合使用,方法引用可以使语言的构造更紧凑简洁,减少冗余代码. ...
- swing桌面四子棋程序开发过程中遇到的一些问题记录(二)
第二个遇到的问题是将JButton按钮设置成透明的按钮.首先UI给我一张透明的图片,如果我直接给Button按钮设置背景图片的话,是没有透明的效果的,只会留下白色的底,设置前后的效果如下图 制作透明的 ...
- 第二章 进程同步(二)——> 重点
2.4 进程同步 2.4.1 进程同步的基本概念 1. 两种形式的制约关系 (1)间接相互制约关系:互斥问题(往往是互斥设备)---是同步的特例 (2)直接相互制约关系:同步问题 注: 互斥问题 ...