第六章:单利模式,懒汉式,饿汉式以及静态内部类,双重检查

第六章:单利模式,懒汉式,饿汉式以及静态内部类,双重检查第六章:单利模式,懒汉式,饿汉式以及静态内部类,双重检查

大家好,又见面了,我是你们的朋友全栈君。


文章转自:https://blog.csdn.net/u012453843/article/details/73743997

单例模式,最常见的就是饥饿模式和懒汉模式,一个直接实例化对象,一个在调用方法时进行实例化对象。

       大家见的最多的莫过于下面这种单例模式了,这种模式是懒汉模式,就是说只有你调用getInstance方法的时候,它才会创建实例。但是这种方式有个非常致命的问题就是在多线程的情况下不能正常工作。

[html] 
view plain
 copy

  1. public class Singleton {    
  2.     private static Singleton instance;    
  3.     private Singleton (){}    
  4.     
  5.     public static Singleton getInstance() {    
  6.        if (instance == null) {    
  7.           instance = new Singleton();    
  8.        }    
  9.        return instance;    
  10.     }    
  11. }   

        懒汉模式要想线程安全,大家第一想到的便是下面这种方式,就是在getInstance方法加上synchronized关键字,但是这种方式也有致命的缺点,那就是并发率太低。

[html] 
view plain
 copy

  1. public class Singleton {    
  2.     private static Singleton instance;    
  3.     private Singleton (){}    
  4.     public static synchronized Singleton getInstance() {    
  5.        if (instance == null) {    
  6.           instance = new Singleton();    
  7.        }    
  8.        return instance;    
  9.     }    
  10. }  

        上面是懒汉模式,下面我们再看下饿汉模式,如下所示。饿汉模式是典型的空间换取时间,当类装载的时候就会创建类的实例,不管你用不用,先创建出来,然后每次调用的时候就不需要再判断了,节省了运行时间。但如果一直没有人调用,这种浪费的空间就不值得,特别是在空间不足的情况下。

[html] 
view plain
 copy

  1. public class Singleton {    
  2.     private static Singleton instance = new Singleton();    
  3.     private Singleton (){}    
  4.     public static Singleton getInstance() {    
  5.        return instance;    
  6.     }    
  7. }   

         在多线程模式中,考虑到性能和线程安全问题,我们一般选择下面两种比较经典的单例模式,在性能提高的同时,又保证了线程安全。


第一种方式:静态内部类


         这种方式是最好的单例模式,而且还是线程安全的,为何这么说呢,静态内部类Singleton在初始化过程中是不会被加载的,只有当用户调用共用的getInstance方法时才会加载内部类Singleton并且实例化Singleton实例,也就是说,静态内部类这种方式也属于懒汉模式,只是实现方式不一样而已。之所以是线程安全的,是因为Singleton是静态的,静态内部类只会被实例化一次,也就是说不管有多少线程,大家拿到的是同一个实例,不会再去进行多次实例化,从而达到了线程安全的目的。由于没有加锁,所以并发性特别高,线程还安全,所以大家以后碰到单例模式,用静态内部类最为合适。如下图所示。

          代码如下:

[html] 
view plain
 copy

  1. package com.internet.singleton;  
  2.   
  3. public class InnerSingleton {  
  4.     private static class Singleton{  
  5.         private static InnerSingleton single = new InnerSingleton();  
  6.     }  
  7.       
  8.     public static InnerSingleton getInstance(){  
  9.         return Singleton.single;  
  10.     }  
  11. }  

第二种方式:双重检查


           首先,我说一下这种设计的初衷是什么,我们看到了getDs()方法中添加了类锁(synchronized (DubbleSingleton.class)),这种锁直接把整个类都锁住了,其它线程访问这个类的任何方法都要排队等候,设计者目的是当第一个线程访问getDs()方法时,它把整个类锁住,然后它把这个类实例化,然后后面的线程进入getDs()方法后一判断发现ds已经不是null了,于是便把第一个线程实例化好的实例返回。这样后续的线程无论并发有多少都没有问题,也不用再排队,直接拿取实例化好的实例即可。既然这样,为何在synchronized (DubbleSingleton.class)当中再判断一次ds是否为null呢?这个其实也好理解,synchronized上面不是模拟了一下初始化对象的准备时间吗?这个模拟是很有必要的,当第一个线程进入getDs()方法后,判断了一下df是否为null,发现是null,然后它就进入初始化对象的准备工作中去了,这个过程可能需要几秒钟,注意这时第一个线程还没有执行到synchronized这一行代码,也就是说还没有加上类锁,在这个过程当中假如又有多个线程要调用getDs()方法,它们便可以同时访问这个方法,这些线程判断ds依然是null,因此会进入到if判断里面,这些线程也进入到初始化对象的准备过程,等到第一个线程初始化对象准备完毕之后,它进入到synchronized这块代码处,给整个类加上了锁,这时后续再有线程的话,就要排队等候调用getDs()了。等第一个线程执行完实例化代码之后,刚才那些趁第一个线程没有锁住类的间隙偷偷摸进来的线程便也都进入到synchronized代码处,这些线程会依次锁住类进入到synchronized代码块中,这时如果代码块中不加一层if判断的话,就会再实例化一次DubbleSingleton类并返回,有几个这样偷溜进来的线程便会实例化几次Dubblesingleton实例。除了这些偷溜进来的线程之外,再来访问的线程由于在类外等待第一个线程执行完之后才有机会进入到方法体中,这时ds早已经实例化过了,因此第一个if判断便不成立,于是直接把当前Dubblesingleton实例返回。也就是说,后面的线程便可以高并发了,不用受锁的限制了。而如果在synchronized锁内再加一层判断的话,由于第二个线程进入锁内时,第一个线程肯定已经执行完了(synchronized这时扮演的是类锁),因此这时ds肯定不是null了,第二个线程一判断,发现ds不为null了,便直接把第一个线程实例化好的实例返回了,同理,其它线程也把第一个线程实例化好的实例返回。从而保证了线程安全。


         双重检查与懒汉加锁模式最大的区别在于,双重检查加锁只是一瞬间的事儿,后续无论有多少个线程都可以自由访问,没有线程安全问题。而懒汉加锁模式由于每个线程都要排队访问getInstance方法,因此效率太低。这就是他们之间的区别。双重检查的代码类如下图所示。


第六章:单利模式,懒汉式,饿汉式以及静态内部类,双重检查

          代码如下:

[html] 
view plain
 copy

  1. package com.internet.singleton;  
  2.   
  3. public class DubbleSingleton {  
  4.       
  5.     private static DubbleSingleton ds;  
  6.       
  7.     public static DubbleSingleton getDs(){  
  8.         if(ds == null){  
  9.             try {  
  10.                 //模拟初始化对象的准备时间…  
  11.                 Thread.sleep(3000);  
  12.             } catch (Exception e) {  
  13.                 e.printStackTrace();  
  14.             }  
  15.             synchronized (DubbleSingleton.class) {  
  16.                 if(ds == null){  
  17.                     ds = new DubbleSingleton();  
  18.                 }  
  19.             }  
  20.         }  
  21.         return ds;  
  22.     }  
  23.       
  24.     public static void main(String[] args){  
  25.         Thread t1 = new Thread(new Runnable() {  
  26.               
  27.             @Override  
  28.             public void run() {  
  29.                 System.out.println(DubbleSingleton.getDs().hashCode());  
  30.             }  
  31.         },”t1″);  
  32.           
  33.         Thread t2 = new Thread(new Runnable() {  
  34.               
  35.             @Override  
  36.             public void run() {  
  37.                 System.out.println(DubbleSingleton.getDs().hashCode());  
  38.             }  
  39.         },”t2″);  
  40.           
  41.                 Thread t3 = new Thread(new Runnable() {  
  42.               
  43.             @Override  
  44.             public void run() {  
  45.                 System.out.println(DubbleSingleton.getDs().hashCode());  
  46.             }  
  47.         },”t3″);  
  48.           
  49.                 t1.start();  
  50.                 t2.start();  
  51.                 t3.start();  
  52.     }  
  53. }  

         运行上面双重检查代码,结果如下图所示,可见,实例是一样的。

第六章:单利模式,懒汉式,饿汉式以及静态内部类,双重检查

         如果把synchronized代码块中的if判断去掉,如下图所示。

第六章:单利模式,懒汉式,饿汉式以及静态内部类,双重检查

          再执行main方法,结果如下图所示,发现三次结果都不一样,说明有线程安全问题。

第六章:单利模式,懒汉式,饿汉式以及静态内部类,双重检查
           综合以上所有情况,可以知道,用静态内部类是最好的单例模式。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/106146.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • Mysql Workbench使用教程

    Mysql Workbench使用教程<1>MySQLWorkbenchMySQLWorkbench为数据库管理员、程序开发者和系统规划师提供可视化的Sql开发、数据库建模、以及数据库管理功能。<2>.MySQLWorkbench的下载和安装(1)安装最新MySql时,有是否安装MySqlWorkbench的选项,可选择安装。(2)可以独立安装MySqlWorkbench…

    2022年6月4日
    41
  • java经典面试题之Spring Boot 面试题汇总附答案(史上最全持续更新)「建议收藏」

    java经典面试题之Spring Boot 面试题汇总附答案(史上最全持续更新)「建议收藏」1.什么是SpringBoot?SpringBoot是Spring开源组织下的子项目,是Spring组件一站式解决方案,主要是简化了使用Spring的难度,简省了繁重的配置,提供了各种启动器,开发者能快速上手。2.SpringBoot有哪些优点?SpringBoot主要有如下优点:容易上手,提升开发效率,为Spring开发提供一个更快、更广泛的入门体验。开箱即用,远离繁琐的配置。提供了一系列大型项目通用的非业务性功能,例如:内嵌服务器、安全管理、运行数据监

    2022年10月12日
    0
  • 【转】值得珍藏的CSS代码集

    【转】值得珍藏的CSS代码集

    2021年8月24日
    58
  • mac双系统选择启动_mac装双系统好不好

    mac双系统选择启动_mac装双系统好不好解决方案1:开机时长按option键,进入系统选择界面:用左右方向键选择到你要设置为默认启动的盘,然后同时按下ctrl+enter键,即可将其设置为默认启动的系统。解决方案2:选择mac系统进入后,点击系统偏好设置—–>启动磁盘—–>进入如下图界面:首先点击最下面的锁图片,然后再进行更改,选择你要设置为默认启动的系统后,点击

    2022年10月6日
    0
  • 云服务器搭建青龙面板每日自动拿京豆

    云服务器搭建青龙面板每日自动拿京豆前言:之前网上有只要扫码一下就可以每天领上百京豆和一些红包的活动,后来呢,扫码就失效了,但是呢,这背后的技术还没有失效。这白嫖活动其实就是用脚本代替我们去参与京东的各种活动,去获取红包和京豆,而这些脚本是部署在电脑上,定时去执行的,接下来,根据网上的大佬的教程,我们也来实现一下。每天100-200京豆不等,坐收渔利,快来试试吧。一、安装前的准备​青龙面板是使用Docker来安装的,理论上,只要有可以运行Docker的电脑都可以进行安装。但是呢,因为脚本要定时运行,所以最好安装在服务器上,或

    2022年10月14日
    0
  • PyCharm入门教程——用户界面导览「建议收藏」

    PyCharm入门教程——用户界面导览「建议收藏」JetBrainsPyCharm是一种PythonIDE,其带有一整套可以帮助用户在使用Python语言开发时提高其效率的工具。此外,该IDE提供了一些高级功能,以用于Django框架下的专业Web开发。通过这篇文章,您可以了解PyCharm用户界面是如何组织的,以帮助您在工作环境中找到自己的方式。当您第一次运行PyCharm或没有打开任何项目时,PyCharm将显示欢迎屏幕,允许快速…

    2022年8月28日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号