美文网首页
单例模式-Java

单例模式-Java

作者: 还是个初学者 | 来源:发表于2018-05-21 18:22 被阅读0次

饿汉、懒汉、双重校验锁及其修正、静态内部类、枚举,线程安全性以及原因

单例模式有以下特点:

1、单例类只能有一个实例。

2、单例类必须自己创建自己的唯一实例。

3、单例类必须给所有其他对象提供这一实例。

饿汉式-立即加载

public class SingleTon {
    private static SingleTon singleTon = new SingleTon();
    private SingleTon(){}
    public static SingleTon getInstance(){
        return singleTon;
    }
}

分析:
类初始化时就进行创建对象,但是不管之后是否使用这个对象,该对象都会占用内存资源。
是线程安全的,可用于多线程
为什么是线程安全的?
由于实例早已加载,多个线程同时访问getInstance时不会出现问题。

懒汉式-延迟加载

public class SingleTon2 {
    private static SingleTon2 singleTon2 = null;
    private SingleTon2(){}
    public SingleTon2 getInstance(){
        if (singleTon2 == null){
            singleTon2 = new SingleTon2();
        }
        return singleTon2;
    }
}

分析:
在需要使用该类实例时才进行加载,能够节省内存,在第一次访问getInstance( )方法时会有些慢(需创建类),之后无影响。
不是线程安全的。
为什么?当singleTon2为null时,如果多个线程同时访问getInstance( ),则都会进入if语句,就会创建多个实例,破坏了单例模式规则。

懒汉式改进版-双重校验锁

public class SingleTon3 {
    private static SingleTon3 singleTon3 = null;
    private SingleTon3(){}
    public static SingleTon3 getInstance(){
        if (singleTon3 == null){
            synchronized (SingleTon3.class){
                if (singleTon3 == null){
                    singleTon3 = new SingleTon3();
                }
            }
        }
        return singleTon3;
    }
}

分析:
改进了懒汉式线程不安全的问题,但是同步还是有点性能消耗。
为什么?当多个线程同时访问getInstance( )方法时,若 singleTon3不为null,则可以直接返回,否则都进入第一个if语句,接下来有一个同步代码块,所以进入if语句的多个线程会“排队”执行,第一个线程进入同步块后,创建了singleTon3实例,所以当接下来的线程都不会执行第二个if语句。
缺点:指令重排问题。

懒汉式改进版-静态内部类

public class SingleTon4 {
    private static class Inner{
        private static final SingleTon4 INSTANCE = new SingleTon4();
    }
    private SingleTon4(){}
    public SingleTon4 getInstance(){
        return Inner.INSTANCE;
    }
}

分析:
延迟加载,线程安全。
分析:
巧妙之处在于静态内部类,内部类(不论是静态内部类还是非静态内部类)都是在第一次使用时才会被加载。 外部类不调用 getInstance()时候内部类是不会加载的,所以实现了延迟加载的功能,并且是线程安全的。

总分析

通过Java反射机制是能够实例化构造方法为private的类的,基本上会使所有的Java单例实现失效。

相关文章

网友评论

      本文标题:单例模式-Java

      本文链接:https://www.haomeiwen.com/subject/yweqjftx.html