zoukankan      html  css  js  c++  java
  • java中常见的单例模式详解

          很多求职者在面试过程中都被问到了单例模式,最常见的问题,比如,每种单例模式的区别是什么?哪些模式是线程安全的?你们项目里用的哪种单例模式?原来没有注意这个问题,回来赶紧打开项目查看了一下代码,才发现我们的项目用到了枚举。有的面试官还会让你手写一种单例模式,我建议大家就写自己项目中用到的那种。下面我就说一说我学到的单例模式。

    一、单例模式的好处

         减少系统资源的消耗。因为这种工具类基本上贯穿程序始终,必然会频繁调用.如果每一次调用都要重新生成实例,那么在内存堆中就会分配相应的内存空间,所以使用单例模式会提高程序的运行速度,减少资源消耗。

    二、单例模式的特点

         1、单例类只能有一个实例:确保某个类只有一个实例

         2、单例类必须自己创建自己的唯一实例:在需要创建单例模式的这个类里,自己new了一个自己的对象

         3、单例类必须给所有其他对象提供这一实例:因为单例类的构造是privite的,所以要给外部提供一个方法来访问这个实例

    三、常见单例模式的使用方法和优缺点

         1、饿汉模式,可用。在初始化的时候就完成实例化了,避免了线程同步的问题,缺点就是无论这个类是否被使用,都会创建一个instance对象,不能延迟加载。如果从始至终都没有用过,造成内存浪费。在类加载的时候就完成了实例化,没有实现延迟加载。

    public class Singleton {  
    //实例化对象 private static Singleton instance = new Singleton();
    //私有构造 private Singleton (){}
    //对外提供一个公共的方法访问这个实例 public static Singleton getInstance() { return instance; } }

         2、饿汉变种,线程安全,可用。将类实例化的过程放在了静态代码块中,也是在类加载的时候,就执行静态代码块中的代码,初始化类的实例。优缺点和上面是一样的。

     public class Singleton {  
          private Singleton instance = null;  
          static {  
          instance = new Singleton();  
          }  
          private Singleton (){}
          public static Singleton getInstance() {  
          return this.instance;  
          }  
     }  

         3、懒汉模式,线程不安全,不推荐用。多线程的时候不能保证实例是唯一的了。比如线程a要使用Singleton,第一次调用发现instance为null,开始创建实例还没完成创建,就在这个时候,CPU去执行线程b了,b发现instance也为null,还会去创建实例。b创建完之后,回来执行a。线程a不会再回去检查instance是否为null了,这样线程a和b各自拥有一个实例,单例失败。但是可以实现延迟加载。

    public class Singleton {  
    //没有final private static Singleton instance;
    //私有构造 private Singleton (){}
    //对外提供一个公共方法来访问这个实例 public static Singleton getInstance() { if (instance == null) { instance = new Singleton(); } return instance; } }

           4、懒汉模式,线程安全。解决饿汉模式带来的问题,就是在创建实例的时候添加一把锁,一个线程必须等待另外一个线程创建完成后才能调用这个方法,这就保证了单例的唯一性了。但是synchronized修饰的同步块要比一般的代码段慢上几倍的,如果多次调用getInstance()这个方法,会影响性能,可以实现延迟加载。

     public class Singleton {  
    //没有final,因为下边要对instance重新赋值 private static Singleton instance;
    //私有构造 private Singleton (){} public static synchronized Singleton getInstance() { if (instance == null) { instance = new Singleton(); } return instance; } }

         5、静态内部类,线程安全,推荐用。实现了延迟加载,减少内部开销,内部类SingletonHolder 只有在getInstance()方法第一次调用的时候才会被加载,内部类加载的时候实现了加载一下INSTANCE 。

    public class Singleton {  
    //内部类实例化INSTANCE  private static class SingletonHolder { private static final Singleton INSTANCE = new Singleton(); } private Singleton (){}
    //对外提供一个公共方法访问INSTANCE  public static final Singleton getInstance() { return SingletonHolder.INSTANCE; } }

         6、双重校验,线程安全,推荐用。既实现线程安全,又能够使性能不受到很大的影响,因为只有在第一次创建实例的时候才会走同步块。

     public class Singleton { 
    //双重校验关键字volatile,被volatile修饰的值,将不会被本地线程缓存,保证所有对该变量的读写都是直接操作共享内存,从而确保多个线程能正确处理该变量,但是使用volatile运行效率并不高。不建议大量使用双重校验。 private volatile static Singleton singleton; private Singleton (){} public static Singleton getSingleton() {
    //第一次校验,是否实例化了 if (singleton == null) {
    //第二次校验,同步块,线程安全的创建实例 synchronized (Singleton.class) { if (singleton == null) { singleton = new Singleton(); } } } return singleton; } }    

    四、单例模式适用的场合

         1、需要频繁的进行创建和销毁的对象

         2、创建对象时耗时过多或耗费资源过多,但又经常用到的对象

         3、工具类对象

         4、频繁访问数据库或文件的对象

      

      

      

  • 相关阅读:
    老齐python-基础7(文件操作、迭代)
    老齐python-基础6(循环 if while for)
    老齐python-基础5(运算符、语句)
    老齐python-基础4(元祖、字典、集合)
    老齐python-基础3(列表)
    老齐python-基础2(字符串)
    Jupyter notebook 的安装、入门
    pycharm 激活码
    VMware无法连接 MKS:套接字连接尝试次数太多正在放弃
    Numpy基本用法简介
  • 原文地址:https://www.cnblogs.com/chentong/p/5604406.html
Copyright © 2011-2022 走看看