前言
大聪明面试记,一步一步成为Offer收割机
在Java面试中,只要涉及到多线程,基本上90%都会问synchronized,一些同学通过背面试题记住了部分知识点,但是对于原理以及底层却不明所以,本文将从用法逐步深入底层实现,让你一次性掌握synchronized的所有知识,成为Offer收割机。
正文
大聪明最近找工作,开场面试官就是一套并发三连:synchronized,volatile,AQS。打得大聪明措手不及,大聪明赶紧回家向自己哥哥大黄牛请教。
那接下来我将重点聊聊,希望给你以借鉴!大家看完觉得还不错的话,别忘了点个赞哦!码字不易
一.使用场景
为了解决多线程并发引起的一系列问题,Java为我们提供了多种并发工具,其中使用synchronized对资源进行加锁是我们最常用的一种。一般使用场景分为以下几种:
可以用来修饰方法(锁的当前实例对象),代码块(需要通过一个对象加锁,锁的括号里面的对象),静态方法(锁整个class对象)
1.修饰方法:此时锁住的是当前实例对象this。
public class SyncTest {
public synchronized void test(){
//do something
}
}
2.修饰代码块:此时锁住的是同步对象obj
public class SyncTest {
public static void test(){
Object obj = new Object();
synchronized (obj){
//do something...
}
}
}
3.修饰静态方法,此时锁的是当前类的class加锁
public class SyncTest {
public static synchronized void test(){
//do something
}
}
二.实现原理
为了探究synchronized是怎么实现的,我写了一个测试代码,大家可以自己拷贝下来尝试通过下面的命令来反编译一下看结果。
- javap -c -p -v xxx.class
测试代码:
public class SyncTest {
Test test = new Test();
public synchronized void test1(){
synchronized (new Integer(1)){
System.out.println("test1");
}
}
}
经过反编译我们可以看到如下结果,
1.在进入同步代码快时,会有 monitorenter 命令,此时会获取锁的所有权。
2.在退出同步代码块时,会有 monitorexit 命令,此时表示解锁操作。
3.在修饰的方法时,通过 ACC_SYNCHRONIZED 标志,此时会隐式的调用 monitorenter 和 monitorexit
4.在第26行,还有一个 monitorexit ,是为了在异常的时候也能正常退出,有一个隐式的try-finally,这个在finally里面调用了 monitorexit。
public synchronized void test1();
descriptor: ()V
flags: ACC_PUBLIC, ACC_SYNCHRONIZED //方法锁的标记位,会隐式调用monitorenter和monitorexit
Code:
stack=3, locals=3, args_size=1
0: new #5
3: dup
4: iconst_1
5: invokespecial #6
8: dup
9: astore_1
10: monitorenter //monitor标记进入
11: getstatic #7
14: ldc #8
16: invokevirtu