1.说一说自己对于 synchronized 关键字的了解
synchronized是解决多线程之间访问资源的同步性,synchronized关键字可以保证被他修饰的资源在任何时刻只有一个线程访问。
在Java6之前,synchronized是重量锁,在其之后官方从JVM层面对synchronized进行了非常大的优化。
2. 说说自己是怎么使用 synchronized 关键字
- 修饰实例方法
- 修饰静态方法
- 修饰代码块
- 对于普通同步方法,锁是当前实例对象。
- 对于静态同步方法,锁是当前类的Class对象。
- 对于同步方法块,锁是Synchonized括号里配置的对象
3. 讲一下 synchronized 关键字的底层原理
首先先写一个synchronized的同步语句块代码,进行编译:
package com.JUC;
/**
* @author xbhog
* @date 2022/2/15
* @apiNote
* @title 双重校验锁实现对象单例(线程安全)
*/
public class Singleton {
private static volatile Singleton uniqueInstance;
public Singleton() {
}
public static Singleton getUniqueInstance(){
//先判断对象是否已经实例过,没有实例化过才进入加锁代码
if(uniqueInstance == null){
//类对象加锁
synchronized (Singleton.class){
if (uniqueInstance == null){
uniqueInstance = new Singleton();
}
}
}
return uniqueInstance;
}
public static void main(String[] args) {
}
}
进入其编译的文件夹中,然后执行分析命令:
javap -c -s -v -l Singleton.class
命令解释:
一般常用的是-v -l -c三个选项。
javap -v classxx,不仅会输出行号、本地变量表信息、反编译汇编代码,还会输出当前类用到的常量池等信息。
javap -l 会输出行号和本地变量表信息。
javap -c 会对当前class字节码进行反编译生成汇编代码。
javap-s 输出内部类型签名
更多内容可以查看该博客:点击
通过上图可以看到:synchronized同步语句块的实现使用是monitorenter和monitorexit指令。其中
monitorenter
指令指向同步代码块的开始位置,monitorexit
指令则指明同步代码块的结束位置。
任意线程对Object(Object由synchronized保护)的访问,首先要获得Object的监视器。如果获取失败(锁的计数器不为 0),线程进入同步队列,线程状态变为BLOCKED。当访问Object的前驱(获得了锁的线程)释放了锁(锁的计数器为0),则该释放操作唤醒阻塞在同步队列中的线程,使其重新尝试对监视器的获取。
具体详情:点击
写一个synchronized的修饰方法代码,进行编译:
package com.JUC;
/**
* @author xbhog
* @date 2022/2/15
* @title: synchronized 修饰方法的情况
*/
public class SynchronizedMethod {
public synchronized void method() {
System.out.println("synchronized 方法");
}
}
synchronized修饰的方法没有monitorenter和monitorexit指令,但是使用的ACC_SYNCHRONIZED
标识,该标识指明了该方法是一个同步方法。
synchronized修饰代码块和修饰方法本质都是使用对象监视器获取的。
4. 谈谈ReentrantLock
- 实现等待可中断
- 可实现公平锁
- 可实现选择性通知:需要与Condition使用,点击进入
详情请看:点击