synchronized
被他修饰的方法或代码块在任一时刻只有一个线程执行。
使用synchronized
修饰实例方法=对实例加锁,执行前需要获得实例的锁
synchronized void method() { //业务代码 }
修饰静态方法 = 对类加锁
synchronized static void method() { //业务代码 }
修饰代码块 = 给指定对象/类加锁
synchronized(this) { //业务代码 } synchronized(类.class){ //blabla }
双重校验锁实现单例模式 线程安全
public class Singleton{ private volatile static Singleton uniqueInstance; private Singleton(){ } public static Singleton getInstance(){ //先判断对象是否已经实例过 if (uniqueInstance == null){ synchronized (Singleton.class){ if (uniqueInstance == null){ uniqueInstance = new Singleton(); } } } } }
上面代码为什么要用volatile
uniqueInstance = new Singleton();的步骤
1.给uni分配内存地址
2.初始化uni
3.把uni指向刚才分配的内存地址
由于jvm指令重排,上述步骤可能按照1 3 2的顺序执行,多线程环境下,执行完3就有调用方get到了uni,但是此时uni还没有被初始化。
volatile可以避免指令重排。
构造方法本身就是线程安全的
不需要用synchronized修饰。
synchronized底层原理
修饰同步语句块
每个对象中都内置了一个objectMonitor对象,获取锁就是获取对象监视器monitor的持有权。
monitorenter指向被synchronized修饰的同步语句块开始的位置,monitorexit指向同步语句块结束的位置。
当执行monitorenter时,会尝试获取对象的锁,如果锁的计数器为0表示可以获取,获取到后计数器会置为1。
monitorexit执行后,计数器会重新置为0。
修饰方法
被修饰的方法会用ACC_SYNCHRONIZED标识,表明该方法是同步方法。本质也是对monitor的获取。
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。