【JAVA设计模式】单例模式之七种常见写法分析

简介

单例模式是一种常见的设计模式,其定义是单例对象的类,在虚拟机中只运行一个实例存在。在多线程环境下,应该提供一定的机制,确保只会产生一个实例

一、饿汉式--静态变量(推荐)

public class MySingleton1 {

    private static MySingleton1 instance = new MySingleton1();

    public static MySingleton1 getInstance() {
        return instance;
    }

    private MySingleton1() {
    }
}

此方法利用了JVM的类加载机制,确保了单例对象只会被实例化一次,保证了线程安全。同时,在MySingleton1类被加载时,就完成了实例化,没有达到懒加载的效果。假设始终没有用法该示例,则会造成浪费内存资源。

二、饿汉式--静态代码块(推荐)

public class MySingleton2 {

    private static MySingleton2 instance;

    static {
        instance = new MySingleton2();
    }

    private MySingleton2() {
    }

    public static MySingleton2 getInstance() {
        return instance;
    }
}

与上一种方式相似,前者利用静态变量,后者利用静态代码块,都是利用JVM的类加载机制,保证只会被实例化一次。

三、懒汉式(不推荐)

public class MySingleton3 {

    private static MySingleton3 instance;

    private MySingleton3() {
    }

    public static MySingleton3 getInstance() {
        if (instance == null) {
            instance = new MySingleton3();
        }
        return instance;
    }
}

在多线程环境下,此方式存在线程安全问题。假设两个线程同时判断instance == null的结果为true,则它们会各自去实例化对象,然后就会产生两个实例,这违背了单例模式的原则。

四、懒汉式--同步方法

public class MySingleton4 {

    private static MySingleton4 instance;

    private MySingleton4() {
    }

    public static synchronized MySingleton4 getInstance() {
        if (instance == null) {
            instance = new MySingleton4();
        }
        return instance;
    }
}

相对于上一种方式,这里给getInstance()方法加上的同步关键字,解决了线程安全,但是也降低了效率,一般不推荐

五、双重检查(推荐)

public class MySingleton5 {

    private static volatile MySingleton5 instance;

    private MySingleton5() {
    }

    public static MySingleton5 getInstance() {

        if (instance == null) {
            synchronized (MySingleton5.class) {
                if (instance == null) {
                    instance = new MySingleton5();
                }
            }
        }
        return instance;
    }
}

与上一种方式相比,同步锁的粒度变小了,大大提高了效率,而且双重检查机制,保证了只会实例化一次

六、内部静态类(推荐)

public class MySingleton6 {

    private MySingleton6() {
    }

    private static class MySingleton6Instance {

        private static final MySingleton6 instance = new MySingleton6();
    }

    public static MySingleton6 getInstance() {
        return MySingleton6Instance.instance;
    }
}

这种实现方式和饿汉式相似,都是利用jvm类加载机制来保证只被实例化一次。与饿汉式不同的是,饿汉式在类加载时完成了实例化,而此种方式下,只会在第一次用到getInstance()方法,才会被实例化。

七、枚举(推荐)

public enum  MySingleton7 {

    RED, BLACK;
}

枚举很简单,上面的RED、BLACK都是单例的,jvm中只会存在一个实例。

单例测试:

public class MySingletonMain {

    public static void main(String[] args) {
        System.out.println(MySingleton1.getInstance() == MySingleton1.getInstance());
        System.out.println(MySingleton2.getInstance() == MySingleton2.getInstance());
        System.out.println(MySingleton3.getInstance() == MySingleton3.getInstance());
        System.out.println(MySingleton4.getInstance() == MySingleton4.getInstance());
        System.out.println(MySingleton5.getInstance() == MySingleton5.getInstance());
        System.out.println(MySingleton6.getInstance() == MySingleton6.getInstance());
        System.out.println(MySingleton7.RED == MySingleton7.RED);
    }
}

单线程环境下,上述所有输出都是true。如果是多线程环境的话,上述提到的线程不安全的实现方式,输出的结果可能会出现false,读者可以自己去实验。就我个人而言,我在实际开发中,一般会采用静态内部类和枚举这两种方式~~~

GitHub地址:https://github.com/ye17186/spring-boot-learn

你可能感兴趣的:(JAVA)