Synchronized关键字
1、对Synchronized的了解
synchronized关键字解决的是多个线程之间访问资源的的同步性。
synchronized可以保证被它修饰的方法或者代码块在任意时刻只能有一个线程执行。
2、使用synchronized
- 修饰实例方法:作用于当前对象实例加锁,进入同步代码前要获得当前对象实例的锁。
- 修饰静态方法: 访问静态 synchronized 方法占用的锁是当前类的锁,而访问非静态 synchronized 方法占用的锁是当前实例对象锁。
- 修饰代码块:指定加锁对象,对给定对象加锁,进入同步代码块前要获得给定对象的锁。
总结: synchronized 关键字加到 static 静态方法和 synchronized(class)代码块上都是是给 Class 类上锁。synchronized 关键字加到实例方法上是给对象实例上锁。尽量不要使用 synchronized(String a) 。因为JVM中,字符串常量池具有缓存功能!
还记得单例模式中的双重校验锁吗?里面就用到了synchronized关键字。
再来回顾一下,争取手写!在看面试题的时候,就有一个面试题让你手写单例模式。
package com.cc.single;
/**
* 双重校验锁实现单例模式
*/
public class Singleton {
//volatile可以禁止指令重排
private volatile static Singleton uniqueInstance;
private Singleton() {
}
public static Singleton getInstance() {
//先判断对象是否已经实例过,没有实例化过才能进入加锁代码
if(uniqueInstance == null) {
//类对象加锁
synchronized (Singleton.class) {
if(uniqueInstance == null) {
uniqueInstance = new Singleton();
}
}
}
return uniqueInstance;
}
}
另外,需要注意 uniqueInstance 采用 volatile 关键字修饰也是很有必要。
uniqueInstance 采用 volatile 关键字修饰也是很有必要的, uniqueInstance = new Singleton(); 这段代码其实是分为三步执行:
- 为 uniqueInstance 分配内存空间
- 初始化 uniqueInstance
- 将 uniqueInstance 指向分配的内存地址
但是由于 JVM 具有指令重排的特性,执行顺序有可能变成 1->3->2。指令重排在单线程环境下不会出现问题,但是在多线程环境下会导致一个线程获得还没有初始化的实例。例如,线程 T1 执行了 1 和 3,此时 T2 调用 getInstance() 后发现 uniqueInstance 不为空,因此返回 uniqueInstance,但此时uniqueInstance 还未被初始化。
使用 volatile 可以禁止 JVM 的指令重排,保证在多线程环境下也能正常运行。