网络网站建设推广,河南省建设工程网站,wordpress 首页函数,酒类产品网站设计synchronized关键字最主要的三种使用方式的总结
修饰实例方法#xff0c;作用于当前对象实例加锁#xff0c;进入同步代码前要获得当前对象实例的锁修饰静态方法#xff0c;作用于当前类对象加锁#xff0c;进入同步代码前要获得当前类对象的锁 。也就是给当前类加锁… synchronized关键字最主要的三种使用方式的总结
修饰实例方法作用于当前对象实例加锁进入同步代码前要获得当前对象实例的锁修饰静态方法作用于当前类对象加锁进入同步代码前要获得当前类对象的锁 。也就是给当前类加锁会作用于类的所有对象实例因为静态成员不属于任何一个实例对象是类成员 static 表明这是该类的一个静态资源不管new了多少个对象只有一份所以对该类的所有对象都加了锁。所以如果一个线程A调用一个实例对象的非静态 synchronized 方法而线程B需要调用这个实例对象所属类的静态 synchronized 方法是允许的不会发生互斥现象因为访问静态 synchronized 方法占用的锁是当前类的锁而访问非静态 synchronized 方法占用的锁是当前实例对象锁。修饰代码块指定加锁对象对给定对象加锁进入同步代码库前要获得给定对象的锁。 和 synchronized 方法一样synchronized(this)代码块也是锁定当前对象的。synchronized 关键字加到 static 静态方法和 synchronized(class)代码块上都是是给 Class 类上锁。这里再提一下synchronized关键字加到非 static 静态方法上是给对象实例上锁。另外需要注意的是尽量不要使用 synchronized(String a) 因为JVM中字符串常量池具有缓冲功能 下面我已一个常见的面试题为例讲解一下 synchronized 关键字的具体使用。
面试中面试官经常会说“单例模式了解吗来给我手写一下给我解释一下双重检验锁方式实现单例模式的原理呗”
双重校验锁实现对象单例线程安全
java public class Singleton {private volatile static Singleton uniqueInstance;private Singleton() {
}public static Singleton getUniqueInstance() {//先判断对象是否已经实例过没有实例化过才进入加锁代码if (uniqueInstance null) {//类对象加锁synchronized (Singleton.class) {if (uniqueInstance null) {uniqueInstance new Singleton();}}}return uniqueInstance;
}
} 另外需要注意 uniqueInstance 采用 volatile 关键字修饰也是很有必要。
uniqueInstance 采用 volatile 关键字修饰也是很有必要的 uniqueInstance new Singleton(); 这段代码其实是分为三步执行
1. 为 uniqueInstance 分配内存空间 2. 初始化 uniqueInstance 3. 将 uniqueInstance 指向分配的内存地址
但是由于 JVM 具有指令重排的特性执行顺序有可能变成 1-3-2。指令重排在单线程环境下不会出现问题但是在多线程环境下会导致一个线程获得还没有初始化的实例。例如线程 T1 执行了 1 和 3此时 T2 调用 getUniqueInstance() 后发现 uniqueInstance 不为空因此返回 uniqueInstance但此时 uniqueInstance 还未被初始化。
使用 volatile 可以禁止 JVM 的指令重排保证在多线程环境下也能正常运行。 参考详细原理Synchronized底层详细原理