`

单例模式的七种写法

阅读更多

       参考:http://www.jianshu.com/p/fdc64df67794

     Java中的单例模式是 一种常见的设计模式,单例模式一共有好多种,这里主要讲解七种,单例模式有以下的特点:

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

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

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

  单例模式确保某个类只有一个实例,而且自行实例化并向整个系统提供这个实例。在计算机系统中,线程池、缓存、日志对象、对话框、打印机、显卡的驱动程序对象常被设计成单例。这些应用都或多或少具有资源管理器的功能。每台计算机可以有若干个打印机,但只能有一个Printer Spooler,以避免两个打印作业同时输出到打印机中。每台计算机可以有若干通信端口,系统应当集中管理这些通信端口,以避免一个通信端口同时被两个请求同时调用。总之,选择单例模式就是为了避免不一致状态,避免政出多头。

  单例模式的好处:避免实例对象的重复创建,减少每次创建对象的时间开销,节约内存空间,避免由于操作多个实例导致的逻辑错误;

  

package offer.Singleton;

/**
 * Created by Taoyongpan on 2017/10/9.
 * 实现单例模式
 * 定义:指实现了特殊模式的类,该类仅能被实例化一次,产生唯一的一个对象
 * 应用举例:windows的任务管理器,回收站,web应用的配置对象,spring中的bean默认也是单例
 * 分类:饿汉式,懒汉式,双检锁,静态内部类,枚举
 * 评价指标有:单例(必须),线程安全,延迟加载,防止反序列化产生新对象,防止反射攻击
 * 实现方法的选择:一般情况下直接使用饿汉式就好了,要求延迟加载时倾向于用静态内部类,涉及到反序列化创建对象或反射问题最好选择枚举
 */
public class Singleton {

    public static void main(String[] args){
        Singleton1 singleton1 = Singleton1.getInstance();
        Singleton2 singleton2 = Singleton2.getInstance();
        Singleton3 singleton3 = Singleton3.getInstance();
        Singleton4 singleton4 = Singleton4.getInstance();
        Singleton5 singleton5 = Singleton5.getInstance();
        Singleton6 singleton6 = Singleton6.getInstance();
        Singleton7 singleton7 = Singleton7.instance;
        singleton7.setAttribute("Taoyongpan");
    }

    /**
     * 版本一:饿汉式
     * 特点:线程安全;在类初始化执行到静态属性时就分配了资源,有资源浪费问题;
     */
    static class Singleton1{
        private static final Singleton1 instance = new Singleton1();
        //或者将私有静态final成员设为公有成员,可省去getInstance公有函数
        private Singleton1(){}
        public static Singleton1 getInstance(){
            return instance;
        }
    }

    /**
     * 版本二:懒汉式(非线程安全)
     * 特点:在第一次调用获取实例方法时分配内存,实现了懒加载;非线程安全;
     */
    static class Singleton2{
        private static Singleton2 instance = null;
        private Singleton2(){}
        public static Singleton2 getInstance(){
            if (instance==null){
                instance = new Singleton2();
            }
            return instance;
        }
    }

    /**
     * 版本三:懒汉式变种(synchronized同步方法,支持多线程)
     * 特点:线程安全;synchronized而造成的阻塞致使效率低,而且很多的阻塞都是没必要的。
     */
    static class Singleton3{
        private static Singleton3 instance = null;
        private Singleton3(){}
        public static synchronized Singleton3 getInstance(){
            if (instance==null){
                instance = new Singleton3();
            }
            return instance;
        }
    }

    /**
     * 版本四:懒汉式变种(synchronized同步块,支持多线程)
     * 特点:写法不同,但与版本三有一样的问题
     */

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

    //版本五:双检锁DCL,支持多线程-懒汉式
    //特点:线程安全;多进行一次if判断,加入volatile修饰,优点是只有在第一次实例化时加锁,之后不会加锁,提升了效率,缺点写法复杂
    //不加入volatile,可能出现第一个if判断不为null,但还并未执行构造函数的情况,因为java编译器会进行指令重排;
    //volatile的两大作用:
    //1防止编译器对被修饰变量相关代码进行指令重排;2读写操作都不会调用工作内存而是直接取主存,保证了内存可见性
    //指令重排:
    //instance = new Singleton5()可主要分为三步:1分配内存,2调用构造函数,3instance指向被分配的内存(此时instance不为null了)
    //正常顺序为123,指令重排可能执行顺序为132,会造成已不为null但未执行构造函数的问题
    //内存可见性:
    //如果字段是被volatile修饰的,Java内存模型将在写操作后插入一个写屏障指令,在读操作前插入一个读屏障指令。
    //这意味着:1一旦完成写入,任何访问这个字段的线程将会得到最新的;2在写入前,任何更新过的数据值是可见的,因为内存屏障会把之前的写入值都刷新到缓存。
    //因此volatile可提供一定的线程安全,但不适用于写操作依赖于当前值的情况,如自增,自减
    //简单来说,volatile适合这种场景:一个变量被多个线程共享,线程直接给这个变量赋值。
    //还能在双检锁上进行优化,引入一个局部变量,但个人觉得效率提成并不大,不再赘述。
    //volatile参考:http://blog.csdn.net/qq_29923439/article/details/51273812
    static class Singleton5{
        private volatile static Singleton5 instance = null;
        private Singleton5(){}
        public  static Singleton5 getInstance(){
            if(instance==null){
                synchronized (Singleton5.class){
                    if(instance==null)
                        instance = new Singleton5();
                }
            }
            return instance;
        }
    }

    //版本六:静态内部类,支持多线程-懒汉式
    //特点:利用静态内部类(只有在出现它的引用时才被加载),完成懒加载;final保证线程安全;
    //类的加载顺序:http://blog.csdn.net/u012123160/article/details/53224469
    //final的作用:
    //1. 在构造函数内对一个final域的写入,与随后把这个被构造对象的引用赋值给一个引用变量,这两个操作之间不能重排序。
    //2. 初次读一个包含final域的对象的引用,与随后读这个final域,这两个操作之间不能重排序。
    //扩展:static变量初始化遵循以下规则:
    //1.静态变量会按照声明的顺序先依次声明并设置为该类型的默认值,但不赋值为初始化的值。
    //2.声明完毕后,再按声明的顺序依次设置为初始化的值,如果没有初始化的值就跳过。
    //static变量初始化参考:http://www.jb51.net/article/86629.htm
    static class Singleton6{
        private Singleton6(){}
        public static Singleton6 getInstance(){
            return Singleton6Holder.instance;
        }
        private static class Singleton6Holder{
            public static final Singleton6 instance = new Singleton6();
        }
    }

    //版本七:通过枚举实现
    //一个完美的单例需要做到:单例,懒加载,线程安全,防止反序列化产生新对象,防止反射攻击
    //而枚举的特性保证了以上除了懒加载以外的所有要求,而且实现代码及其简单
    //Enum的单例模式参考:http://www.jianshu.com/p/83f7958b0944
    enum Singleton7{
        instance;
        private String attribute;
        void setAttribute(String attribute){
            this.attribute = attribute;
        }
        String getAttribute(){
            return this.attribute;
        }
    }

}

 

  

分享到:
评论

相关推荐

    Java设计模式之单例模式的七种写法

    Java设计模式之单例模式的七种写法 单例模式是一种常见的设计模式,它确保某个类只有一个实例,而且自行实例化并向整个系统提供这个实例。在计算机系统中,线程池、缓存、日志对象、对话框、打印机的驱动程序对象常...

    单例模式的八种写法比较

    android资料 单例模式的八种写法比较 单例模式是一种常用的软件设计模式,其定义是单例对象的类只能允许一个实例存在。

    java-单例模式几种写法

    单例模式是软件设计模式中的一种,用于控制类的实例化过程,确保一个类只有一个实例,并提供全局访问点。在Java中,实现单例模式有多种方法,每种方法都有其特点和适用场景。以下是对这六种常见单例模式实现方式的...

    Java:单例模式的七种写法

    第七种(双重检查锁定DCL,即double-checked locking):1 public class Singleton {2 private volatile static Singleton singleton;3 private Singleton (){}4 public static Singleton getInstance() {5 if ...

    java单例模式开发的7种写法

    ### Java单例模式开发的七种写法 #### 概述 单例模式是一种常用的软件设计模式,其目的是确保一个类仅有一个实例,并提供一个全局访问点。在Java编程语言中,实现单例模式的方法有很多种,不同的实现方式具有不同的...

    单例的多种写法和说明比较

    单例模式是软件设计模式中的一种,用于控制类的实例化过程,确保一个类在整个程序运行期间只有一个实例存在。在Java、C#等面向对象语言中,单例模式被广泛应用,尤其是在需要频繁创建和销毁的对象,或者需要全局访问...

    Tom_20170324_Java设计模式之单例模式的七种写法1

    2、单例类必须自己创建自己的唯一实例 3、单例类必须给所有其他对象提供这一实例 2、资源加载和性能:饿汉式在类创建的同时就实例化一个静态对象出来,不管之后会不会

    unity中涉及的三种单例模式

    在Unity游戏开发中,单例模式是一种常用的编程设计模式,它确保一个类只有一个实例,并提供一个全局访问点。这种模式在处理需要跨场景共享数据的情况时尤其有用,因为Unity的场景切换可能导致对象被销毁,而单例则...

    JavaSE单例模式各种写法.doc

    每种单例模式都有其适用场景: - **懒汉式(线程不安全)**适用于简单的、非并发环境。 - **懒汉式(线程安全)**在需要保证线程安全的情况下使用,但性能较低。 - **饿汉式**适用于对性能要求较高的情况,不支持...

    设计模式_创建型_单例模式.md

    因此,枚举的写法也是实现单例模式的最佳实践。 ## 单例模式可能遇到的问题 ### 反序列化破坏单例 当单例对象实现java.io.Serializable接口后,即使单例类实现了readResolve()方法,但反序列化过程中仍然会创建一...

    IOS 中两种单例模式的写法实例详解

    在iOS开发中,单例模式是一种常见的设计模式,它确保一个类只有一个实例,并提供一个全局访问点。在Objective-C中,有两种官方推荐的单例模式实现方式,这两种方法都保证了单例对象的线程安全。以下是这两种写法的...

    01 设计模式之单例模式.pdf

    双重检查锁定是一种优化懒汉式单例模式的写法,它可以在多线程环境中保证实例的唯一性,同时避免不必要的同步开销。通过在声明实例变量时加上volatile关键字,可以确保在读取该变量时不会发生指令重排序,从而保证了...

    objective-c单例模式的完整书写方式

    Objective-C中的单例模式是一种设计模式,用于在整个应用程序中确保只有一个特定类的实例存在,并提供一个全局访问点来获取这个实例。单例模式在iOS开发中广泛应用,特别是在管理共享资源、配置设置或网络请求等场景...

    java单例模式详解Java系列2021.pdf

    综上所述,单例模式在Java中的实现有多种方式,每种方式都有它的适用场景和潜在风险。开发者需要根据具体需求和环境选择合适的实现方式,并注意可能出现的问题,如线程安全、内存可见性和反射攻击等,来保证单例模式...

Global site tag (gtag.js) - Google Analytics