page contents

如何使用synchronized实现单例?有那些注意点?

轩辕小不懂 发布于 2022-04-15 14:51
阅读 385
收藏 0
分类:Java开发

最佳答案 2022-04-25 16:13

3481
Nen
Nen
- 程序员

可以通过双重校验锁实现对象单例,代码如下:

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 = new Singleton(); 这段代码其实是分为三步执行:

    为 uniqueInstance 分配内存空间

    初始化 uniqueInstance

    将 uniqueInstance 指向分配的内存地址

但是由于 JVM 具有指令重排的特性,执行顺序有可能变成 1->3->2。指令重排在单线程环境下不会出现问题,但是在多线程环境下会导致一个线程获得还没有初始化的实例。例如,线程 T1 执行了 1 和 3,此时 T2 调用 getUniqueInstance() 后发现 uniqueInstance 不为空,因此返回 uniqueInstance,但此时 uniqueInstance 还未被初始化。

使用 volatile 可以禁止 JVM 的指令重排,保证在多线程环境下也能正常运行。

请先 登录 后评论