设计模式笔记(一):Singleton 设计模式

今天开始学习设计模式,借此机会学习并整理学习笔记。

设计模式是一门不区分语言的课程,什么样的编程语言都可以用到设计模式。如果说java语法规则比作武功招式的话,那么设计模式就是心法。

设计模式共有23种,常见的19种,最常用的9-10种。

设计模式分三种类型:创建型、结构型、行为型;

其中创建型包含单例设计模式、工厂模式、抽象工厂模式、原型模式、建造者模式;结构型包含代理模式、装饰器模式、适配器模式、外观模式、组合模式、享元模式、桥梁模式;行为型包含:策略模式、责任链模式、命令模式、中介者模式、模板方法模式、迭代器模式、访问者模式、观察者模式、解释器模式、备忘录模式、状态模式。

好了,凑字完毕,开始今天的学习整理

Singleton设计模式,就是单例设计模式。

单例设计模式有句顺口溜:单例一实例,私有构造器

单例设计模式分为饿汉式和懒汉式。

饿汉式是用户没有请求你的要求时,已经把这个实例提前创建出来了;

懒汉式则是需只有外部需要调用getInstance的时候,才回去初始化这个单例。

这样就有一个区别:

饿汉式:用空间换时间

懒汉式:用时间换空间

下面是具体代码实现:

singleton 单例设计模式
 1:饿汉式
饿汉式优缺点:
优点: 实现简单,没有多线程同步问题(默认条件下是线程安全的)
缺点:当类加载SingletonTest的时候,会初始化static的instance,静态变量被创建并分配内存空间,从此,这个static的instance对象便一直占着此段内存(即使你并没有使用该实例,当类被卸载时,静态变量被摧毁,并释放所占有的内存,因此在某些特定条件下会耗费内存。
简单来说 用空间换时间

为什么在饿汉式中不讨论线程安全问题:
 因为类里面的静态实例都是由类加载器来负责初始化的,类加载器classLoader在出现类名的时候就会开始工作,把类的静态实例全部初始化,而且只初始化一次,所以一定是安全的,

public class Singleton {
    //静态一实例  将自身实例化对象设置为一个属性,并用static final修饰
    private static Singleton instance  = new Singleton();

//私有构造器
    private Singleton() {
        
    }
    
    //外部通过静态方法返回该实例
    public static Singleton getInstance() {
        return instance;
    }     
}

2:懒汉式
懒汉式优缺点:
优点:实现起来比较简单,当类SingletonTest被加载的时候,静态变量static的instance未被创建并分配内存空间,当getInstance()方法第一次调用的时,初始化instance变量,并分配内存空间。在某些条件下这种方式会节省内存。

缺点:在多线程环境中这种方法是完全错误的,根本不能保证单例的状态 需要添加synchronized(锁)

public class Singleton2 {
    //对象赋予null值 或者 不赋值  
    private static Singleton2 instance2;

//私有构造器
    prviate Singleton2() {
        
    }
    
    //外部获取实例对象  此时为了线程安全 需要使用synchronized(锁)
    public static synchronized Singleton2 getInstance2() {
        if(instance2 == null) {
            instance2 = new Singleton2();
        }
        return instance2;
    }       
}

3:如何兼顾饿汉式与懒汉式的优点:
 第一种:使用静态内部类的方法

优点:外部类加载时不予要立即加载内部类,内部类不被加载就不会初始化instance 故而占用内存。当Singleton 第一次加载时,并不需要去加载Singleton3handler只有当getInstance()方法第一次被 调用时,才会去初始化Instance,第一次调用getInstance()方法会导致虚拟机加载在SingletonHangdler类,这种方法不仅能确保线程安全,也能保证单例的唯一性,同时也延迟单例的实例化,

public class Singleton3 {
        //构建静态内部类
        private static class Singleton3Handler{
            private static Singleton3 instance3 = new Singleton3();
        }
        //私有构造器
        private  Singleton3() {
            
        }
        //外部获取对象
        public static Singleton3 getInstance() {
            return Singleton3.Singleton3Handler.instance3;
        }
}

第二种:使用枚举方式  
优点: 线程是安全的

public enum Singleton4{
        INSTANCE;
        public void method() {
            //TODO
        }
    }
 在任何情况下,它都是一个单例 我们可以直接 Singleton4.INSTANCE 引用

第三种:使用DCL模式     需要使用Volatile 关键字
双重锁懒汉式( Double Check Lock  简称 DCL )
优点:只有对象需要被使用的时候才创建,第一次判断instance21 == null 为了避免非必要枷锁,当第一次加载时才对实例进行枷锁在实例化。这样就可以节约内存空间,有可以保证线程安全。

缺点:但是,由于jvm存在乱序执行功能,DCL也会出现线程不安全的情况。

比如:  
      instance21  = new Singleton2;
      这个步骤,其实在jvm里面的执行分为三步:
      1:在堆内开辟内存空间
      2;在堆内存中实例化Singleton2里面的各个参数
      3:  把对象指向堆内存
      但是这个缺点在jdk1.6以后 只需要定义 为: private volatile  static Singleton5 instance5 = null;  就可以
      解决此问题  。  volatile确保instance每次在均在主内存中读取,这样虽然会牺牲一点效率。

public class Singleton5{
    // 对象一实例
    private volatile static Singleton5 instance5;
    //私有构造器
    private Singleton5() {
        
    }
    //DCL懒汉式
    public static Singleton5 getInstance() {
        if(instance5==null) {
            synchronized(Singleton5.class){
                if(instance5 == null) {
                    instance5 = new Singleton5();
                }
            }
        }
        return instance5;
    }
}

需要注意的是:

  需要延迟加载的情况下使用第一、第二种;

  需要防止序列化问题、反射攻击使用第三种。

才疏学浅,如有错误,恳请指教!

(0)

相关推荐

  • java常见设计模式之---单例模式

    java常见设计模式之---单例模式 1.单例模式简介 应用场景举例 2.单例模式的特点 3.单例模式和静态类 4.单例模式的经典实现 饿汉式单例(典型实现) 饿汉式-静态代码块 懒汉式单例创建,五种 ...

  • 图解Java设计模式之单例设计模式

    图解Java设计模式之单例设计模式 设计模式介绍 设计模式类型 单例设计模式介绍 饿汉式(静态常量) 饿汉式(静态代码块) 懒汉式(线程不安全) 懒汉式(线程安全,同步方法) 懒汉式(线程安全,同步代 ...

  • 设计模式:单例模式 (关于饿汉式和懒汉式)

    定义 单例模式是比较常见的一种设计模式,目的是保证一个类只能有一个实例,而且自行实例化并向整个系统提供这个实例,避免频繁创建对象,节约内存. 单例模式的应用场景很多, 比如我们电脑的操作系统的回收站就 ...

  • 单例模式的八种写法

    单例模式作为日常开发中最常用的设计模式之一,是最基础的设计模式,也是最需要熟练掌握的设计模式.单例模式的定义是:保证一个类仅有一个实例,并提供一个访问它的全局访问点.那么你知道单例模式有多少种实现方式 ...

  • 设计模式之单例模式(Singleton Pattern)

    一.定义 一个类只有一个实例,且该类能自行创建这个实例的一种模式. 二.单例模式举例 例如,Windows 中只能打开一个任务管理器,这样可以避免因打开多个任务管理器窗口而造成内存资源的浪费,或出现各 ...

  • 【设计模式】单例模式(Singleton Pattern)

    懒汉式 public class Singleton { private static Singleton instance; private Singleton() {}; public stati ...

  • 【转】C#设计模式-单例模式(Singleton Pattern)

    目录 介绍 第一个版本 --不是线程安全的 第二个版本 -- 简单的线程安全 第三个版本 - 使用双重检查锁定尝试线程安全 第四个版本 - 不太懒,不使用锁且线程安全 第五版 - 完全懒惰的实例化 第 ...

  • [读书笔记] 《大话设计模式》

    作者:程杰 前言 成为 诗人 后 可能 不需要 刻意 地 按照 某种 模式 去 创作, 但 成为 诗人 前 他们 一定 是 认真 地 研究 过 成百上千 的 唐诗 宋词. 古今 名句. 第 1 章 代 ...

  • 大话设计模式笔记(十八)の单例模式

    单例模式 定义 保证一个类仅有一个实例,并提供一个访问它的全局访问点. 通常我们可以让一个全局变量使得一个对象被访问,但它不能防止你实例化多个对象.一个最好的办法就是,让类自身负责保存它的唯一实例.这 ...

  • 大话设计模式笔记(十七)の迭代器模式

    迭代器模式 定义 提供一种方法顺序访问一个聚合对象中各个元素,而又不暴露该对象的内部表示. 什么时候用? 当你需要访问一个聚集对象,而且不管这些对象是什么都需要遍历的时候,你就应该考虑用迭代器模式. ...

  • 大话设计模式笔记(十三)の状态模式

    举个栗子 问题描述 上班的日子,上午状态好,中午想睡觉,下午渐恢复,加班苦煎熬.根据时间的不同体现不同的工作状态. 简单实现 Work /** * 工作类 * Created by callmeDev ...

  • 大话设计模式笔记(七)の原型模式

    举个栗子 问题描述 要求有一个简历类,必须要有姓名,可以设置性别和年龄,可以设置工作经历,最终需要三份简历. 简单实现 简历类 /** * 简历类 * Created by callmeDevil o ...

  • 大话设计模式笔记(六)の工厂方法模式

    栗子回顾 简单工厂模式: https://www.cnblogs.com/call-me-devil/p/10926633.html 运算类使用工厂方法模式实现 UML图 代码实现 工厂接口 /** ...