1、了解Singleton模式

程序在运行时,通常都会生成很多实例。例如,表示字符串的java . lang . string类的实例与字符串是- -对- -的关系,所以当有1000个字符串的时候,会生成1000个实例。但是,当我们想在程序中表示某个东西只会存在-一个时,就会有“只能创建-一个实例” 的需求。典型的例子有表示程序所运行于的那台计算机的类、表示软件系统相关设置的类等。Singleton 是指只含有一-个元素的集合。因为本模式只能生成-一个实例,因此以Singleton命名。

意图:保证一个类仅有一个实例,并提供一个访问它的全局访问点。

主要解决:一个全局使用的类频繁地创建与销毁。

何时使用:当您想控制实例数目,节省系统资源的时候。

如何解决:判断系统是否已经有这个单例,如果有则返回,如果没有则创建。

关键代码:构造函数是私有的。

单例模式在Java中的实现方式:

懒汉式,饿汉式,静态内部类,枚举等

2、懒汉式以及双检锁

getInstance()为简单的懒汉式方式,在多线程是不安全的。

getInstance2()为双检锁,又叫双重校验锁。既保证了线程安全,又比直接上锁提高了执行效率,还节省了内存空间。推荐使用

package cn.design.single;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException; /**
* @author by lin
* @Classname LazyMan
* @Description 懒汉式
* @Date 2020/4/12 16:36
* @Created in 2020/4/12 16:36 by lin
*/
public class LazyMan {
   //增加标志位 , 防止反射 破坏
   private static boolean flag = false;    // 保证私有,只有一个
   private LazyMan() {
       synchronized (LazyMan.class) {
           if (!flag) {
               flag = true;
          } else {
               throw new RuntimeException("不要试图使用反射破坏 异常");
          }
      }
  }    // 避免指令重排
   private volatile static LazyMan lazyMan = null;    //多线程下 是不安全的
   public static LazyMan getInstance() {
       if (lazyMan == null) {
           lazyMan = new LazyMan();
      }
       return lazyMan;
  }    // 解决 双检锁
   public static LazyMan getInstance2() {
       if (lazyMan == null) {
           synchronized (LazyMan.class) {
               if (lazyMan == null) {
                   // 并不是一个原子操作
                   lazyMan = new LazyMan();
                   // 可能会出现指令重排
                   /*
                    * 执行流程
                    * 1、分配内存空间
                    * 2、执行构造方法,初始化对象
                    * 3、把这个对象执行这个空间
                    *
                    * 123 正常执行
                    * 132 指令重排
                    *     A 线程 执行到了3
                    *     B 线程进来,此时 lazyMan 未执行 new 初始化操作2, 发生了指令重排, 此地址空间下的对象为空 , 不安全
                    *     添加 volatile ,避免 指令重排
                    */
              }
          }
      }
       return lazyMan;
  }    // 反射 对象被破坏 破坏单例
   public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException, NoSuchFieldException {
       Field flag = LazyMan.class.getDeclaredField("flag");
       flag.setAccessible(true);
       Constructor<LazyMan> declaredConstructor = LazyMan.class.getDeclaredConstructor();
       declaredConstructor.setAccessible(true);
       LazyMan instance1 = declaredConstructor.newInstance();
       System.out.println(instance1);
       // 道高一尺魔高一丈 修改 flag的值
       flag.set(instance1, false);
//       LazyMan instance = LazyMan.getInstance2();
//       System.out.println(instance);
       Constructor<LazyMan> declaredConstructor2 = LazyMan.class.getDeclaredConstructor();
       declaredConstructor2.setAccessible(true);
       LazyMan instance3 = declaredConstructor2.newInstance();
       System.out.println(instance3);
  } }

2、饿汉式

package cn.design.single;

/**
* @author by lin
* @Classname HungryMan
* @Description 饿汉式
* @Date 2020/4/12 16:48
* @Created in 2020/4/12 16:48 by lin
*/
public class HungryMan {    private HungryMan() {
  } //   private byte[] byte=new byte[1024];    private static final HungryMan hungryMan = new HungryMan();    public static HungryMan getInstance() {
       return hungryMan;
  }
   // 问题: 如果此类内部 有很多的变量 会造成性能问题
   // byte 很多的 变量 }

3、静态内部类

package cn.design.single;

/**
* @author by lin
* @Classname Holder
* @Description 静态内部类
* @Date 2020/4/12 16:50
* @Created in 2020/4/12 16:50 by lin
*/
public class Holder {
   private Holder(){}    public static Holder getInstance(){
       return InnerClass.HOLDER;
  }    public static class InnerClass{
       private static final Holder HOLDER = new Holder();   } }

4、最完美的方式 枚举类

package cn.design.single;

import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException; /**
* @author by lin
* @Classname EnumSingle
* @Description TODO
* @Date 2020/4/12 18:12
* @Created in 2020/4/12 18:12 by lin
*/ // 1.5 以后出来的
// enum 本身也是一个 class 类 public enum EnumSingle {
  INSTANCE;   public EnumSingle getInstance() {
      return INSTANCE;
  }
}
class test{
  public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
      EnumSingle instance1 = EnumSingle.INSTANCE;
      EnumSingle instance2 = EnumSingle.INSTANCE;
      System.out.println("instance1 = " + instance1);
      System.out.println("instance2 = " + instance2);
      // 使用反射 破坏尝试
      Constructor<EnumSingle> declaredConstructor = EnumSingle.class.getDeclaredConstructor(String.class, int.class);
      declaredConstructor.setAccessible(true);
      EnumSingle newInstance = declaredConstructor.newInstance();
      System.out.println(newInstance);       /* 问题分析
        * 枚举类底层其实也是class, 集成了 Enum
        * idea 编译结果 和 javap -p xxx.class 反编译,也是 无惨的私有构造, 欺骗了我们
        * 使用jad 工具, 底层其实是 有参的 私有构造 参数为 String , int 俩个
        * 结果得到了 我们的 异常 Cannot reflectively create enum objects
        * declaredConstructor.newInstance(); 源码分析 为 : 为枚举类的时候, 不能create enum objects
        */
  }
}

5、小结

上面就是单例模式的五种主要写法。我们来总结下,一般情况下,懒汉式(包含线程安全和线程不安全方式)都比较少用;饿汉式和双检锁都可以使用,可根据具体情况自主选择;在要明确实现 lazy loading 效果时,可以考虑静态内部类的实现方式;若涉及到反序列化创建对象时,大家也可以尝试使用枚举方式。

要想实现效率高的线程安全的单例,我们必须注意以下两点:

  • 尽量减少同步块的作用域;

  • 尽量使用细粒度的锁。

发哥讲

如果你觉得文章还不错,就请点击右上角选择发送给朋友或者转发到朋友圈~

● 扫码关注公众号,  转载请备注来源,和链接

6、单例模式 Singleton模式 只有一个实例 创建型模式的更多相关文章

  1. ANDROID 中设计模式的採用--创建型模式

     所谓模式就是在某一情景下解决某个问题的固定解决方式. 全部的创建型模式都是用作对象的创建或实例化的解决方式. 1 简单工厂模式 创建对象的最简单方法是使用new来创建一个对象,假设仅仅创建一种固 ...

  2. Java 23种设计模式详尽分析与实例解析之一--创建型模式

    面向对象的设计原则 常用的面向对象设计原则包括7个,这些原则并不是独立存在的,它们相互依赖.互为补充. Java设计模式 创建型模式 简单工厂模式 模式动机: 考虑一个简单的软件应用场景,一个软件系统 ...

  3. 设计模式之美:Creational Patterns(创建型模式)

    创建型模式(Creational Patterns)抽象了对象实例化过程. 它们帮助一个系统独立于如何创建.组合和表示它的那些对象. 一个类创建型模式使用继承改变被实例化的类. 一个对象创建型模式将实 ...

  4. .NET设计模式(7):创建型模式专题总结(Creational Pattern)(转)

    概述 创建型模式,就是用来创建对象的模式,抽象了实例化的过程.它帮助一个系统独立于如何创建.组合和表示它的那些对象.本文对五种常用创建型模式进行了比较,通过一个游戏开发场景的例子来说该如何使用创建型模 ...

  5. .NET设计模式(7):创建型模式专题总结(Creational Pattern)

    ):创建型模式专题总结(Creational Pattern)    创建型模式专题总结(Creational Pattern) --.NET设计模式系列之七 Terrylee,2006年1月 转载: ...

  6. NET设计模式 第二部分 创建型模式(6):创建型模式专题总结(Creational Pattern)

    创建型模式专题总结(Creational Pattern) ——.NET设计模式系列之七 Terrylee,2006年1月 概述 创建型模式,就是用来创建对象的模式,抽象了实例化的过程.它帮助一个系统 ...

  7. 设计模式GOF23(创建型模式)

    • 创建型模式:  单例模式.工厂模式.抽象工厂模式.建造者模式.原型模式.   • 结构型模式: –适配器模式.桥接模式.装饰模式.组合模式.外观模式.享元模式.代理模式.   • 行为型模式: 模 ...

  8. 工厂方法模式——创建型模式02

    1. 简单工厂模式     在介绍工厂方法模式之前,先介绍一下简单工厂模式.虽然简单工厂模式不属于GoF 23种设计模式,但通常将它作为学习其他工厂模式的入门,并且在实际开发中使用的也较为频繁. (1 ...

  9. 设计模式学习之简单工厂(Simple Factory,创建型模式)(1)

    简单工厂(Simple Factory,创建型模式) 第一步: 比如我们要采集苹果和香蕉,那么我们需要创建一个Apple类和Banana类,里面各自有采集方法get(),然后通过main方法进行调用, ...

随机推荐

  1. redis(二十四):Redis分布式锁以及实现(python)

    阅读目录 什么事分布式锁 基于redis实现分布式锁 一.什么是分布式锁 我们在开发应用的时候,如果需要对某一个共享变量进行多线程同步访问的时候,可以使用我们学到的锁进行处理,并且可以完美的运行,毫无 ...

  2. mysql子查询习题98

    #1.查询工资最低的员工信息:last name, salary SELECT last_name, salary FROM employees WHERE salary = ( SELECT MIN ...

  3. ASP.NET Core3.1使用Identity Server4建立Authorization Server-2

    前言 建立Web Api项目 在同一个解决方案下建立一个Web Api项目IdentityServer4.WebApi,然后修改Web Api的launchSettings.json.参考第一节,当然 ...

  4. springboot --- 之SSM框架整合

    1.pom依赖: 即:spring-boot的基本jar ---- 内置springmvc和spring Thymeleaf jar 热部署 jar ---方便二次加载 ctrl+f9再次编译 Myb ...

  5. 机器学习 | SVD矩阵分解算法,对矩阵做拆分,然后呢?

    本文始发于个人公众号:TechFlow,原创不易,求个关注 今天是机器学习专题第28篇文章,我们来聊聊SVD算法. SVD的英文全称是Singular Value Decomposition,翻译过来 ...

  6. .NET 使用sock5做代理(不是搭建服务端)

    在日常开发中经常会遇到这些需求,爬取数据,都知道现在通常用python爬取是很方便的,但是免不了还有很多伙伴在用NET来爬取,在爬取数据的时候我们知道需要使用代理服务器,如果不用代理,你的IP很有可能 ...

  7. P4547 [THUWC2017]随机二分图(状压,期望DP)

    期望好题. 发现 \(n\) 非常小,应该要想到状压的. 我们可以先只考虑 0 操作. 最难的还是状态: 我们用 \(S\) 表示左部点有哪些点已经有对应点, \(T\) 表示右部点有哪些点已经有对应 ...

  8. 利用python实现平稳时间序列的建模方式

    一.平稳序列建模步骤 假如某个观察值序列通过序列预处理可以判定为平稳非白噪声序列,就可以利用ARMA模型对该序列进行建模.建模的基本步骤如下: (1)求出该观察值序列的样本自相关系数(ACF)和样本偏 ...

  9. Docker 入门教程(3)——Dockerfile

    Dockerfile Dockerfile是一个文本文件,用来定制镜像. 镜像是分层存储的,前一层会是下一层的基础.而镜像的定制就是定制每一层镜像在上一层做了什么改变. Dockerfile其内包含一 ...

  10. linux虚拟机正常安装完成后获取不到IP的解决办法-网卡

    通常正常情况下安装完linux虚拟机,只需要使用桥接并修改配置文件/etc/sysconfig/network-scripts/ifcfg-eth0,将如下参数值改为如下: ONBOOT=yes NM ...