热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

101单例模式(创建型)

1单例介绍定义:Ensureaclasshasonlyoneinstance,andprovideaglobalpointofaccesstoit.(

1单例介绍

定义:Ensure a class has only one instance, and provide a global point of access to it.

(确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例。)

概念:一个对象只能创建一个实例

遵循规则:私有化构造器,通过其他方法获取该类的唯一实例

要点:

①一个类只能有一个实例构造器私有化

②必须自行创建这个实例含有一个该类的静态变量来保存这个唯一的实例

③必须自行向整个系统提供这个实例

对外提供获取该实例对象的方式:①直接暴露②用静态变量的get方法获取

单例模式的几种

1饿汉式(静态常量)

2 饿汉式(静态代码块)

3懒汉式(线程不安全)

4懒汉式(线程安全,同步方法,同步代码块,同步锁)

5 双重检查

6 静态内部类

7枚举


2示例代码


2.1饿汉式


2.1.1饿汉式-静态常量

饿汉式:根据jvm虚拟机中:静态修饰的只会加载一次来实现单例效果

class SingletonHungry1 {//饿汉式 静态常量/*/优势 简单 避免多线程的同步问题劣势 没有达到懒加载的效果 内存的浪费*/private SingletonHungry1(){System.out.println("SingletonHungry1静态常量构造方法");}private final static SingletonHungry1 INSTANCE = new SingletonHungry1();public static SingletonHungry1 getInstance(){return instance;}
}
//测试是否满足单例public static void main(String[] args) {SingletonHungry2 singletonHungry1 = SingletonHungry2.getInstance();SingletonHungry2 singleton2 = SingletonHungry2.getInstance();System.out.println(singletonHungry1==singleton2);}

2.1.2饿汉式-静态代码块

class SingletonHungry2 {//饿汉式 静态代码块/*/优势 简单 避免多线程的同步问题劣势 没有达到懒加载的效果 内存的浪费*/static {instance = new SingletonHungry2();}private static SingletonHungry2 instance =null;private SingletonHungry2(){System.out.println("SingletonHungry2静态常量构造方法");}public static SingletonHungry2 getInstance(){return instance;}
}

优点:这种写法比较简单,就是在类装载的时候就完成实例化。避免了线程同步问题。

缺点:在类装载的时候就完成实例化,没有达到Lazy Loading的效果。如果从始至终从未使用过这个实例,则会造成内存的浪费

这种方式基于classloder机制避免了多线程的同步问题,不过,instance在类装载时就实例化,在单例模式中大多数都是调用getInstance方法, 但是导致类装载的原因有很多种,因此不能确定有其他的方式(或者其他的静态方法)导致类装载,这时候初始化instance就没有达到lazy loading的效果

结论:这种单例模式可用,可能造成内存浪费


2.1.3枚举(推荐)饿汉式

public enum EnumSingle {INSTANCE; //属性public void sayOK() {System.out.println("ok~");}
}public class SingletonTest {public static void main(String[] args) {for (int i &#61; 0; i <10; i&#43;&#43;) {new Thread(()->{EnumSingle enumSingle1 &#61; EnumSingle.INSTANCE;EnumSingle enumSingle2 &#61; EnumSingle.INSTANCE;System.out.println(enumSingle1&#61;&#61;enumSingle2);},String.valueOf(i)).start();}}
}

这借助JDK1.5中添加的枚举来实现单例模式。不仅能避免多线程同步问题&#xff0c;而且还能防止反序列化重新创建新的对象。

这种方式是Effective Java作者Josh Bloch 提倡的方式

 结论&#xff1a;推荐使用    


2.2懒汉式


2.2.1懒汉式(线程不安全) 

不推荐使用

class SingletonLazy1 {//懒汉式 线程不安全/*优势&#xff1a;起到了懒加载的效果 不会造成内存浪费劣势&#xff1a;线程不安全 不推荐这种方式的*/private SingletonLazy1(){System.out.println("SingletonLazy1懒汉式构造方法");}private static SingletonLazy1 instance &#61; null;public static SingletonLazy1 getInstance(){if (instance&#61;&#61;null){instance &#61; new SingletonLazy1();}return instance;}
}
public class SingletonLazyTest {//在并发情况下&#xff0c;不满足单例效果public static void main(String[] args) {for (int i &#61; 0; i <10; i&#43;&#43;) {new Thread(()->{SingletonLazy1 singletonLazy1 &#61; SingletonLazy1.getInstance();System.out.println(singletonLazy1);},String.valueOf(i)).start();}}
}

查看运行结果&#xff0c;在并发情况下&#xff0c;会创建多个对象&#xff0c;构造方法会执行多次

起到了Lazy Loading的效果&#xff0c;但是只能在单线程下使用。

如果在多线程下&#xff0c;一个线程进入了if (singleton &#61;&#61; null)判断语句块&#xff0c;还未来得及往下执行&#xff0c;另一个线程也通过了这个判断语句&#xff0c;这时便会产生多个实例。所以在多线程环境下不可使用这种方式

结论&#xff1a;在实际开发中&#xff0c;不要使用这种方式.


2.2.2懒汉式(线程安全&#xff0c;同步方法&#xff0c;同步代码块&#xff0c;同步锁)

不推荐使用

class SingletonLazy2 {//懒汉式 线程安全 同步锁 三种方式/*解决了线程安全问题&#xff0c;但是效率太低*/private static SingletonLazy2 instance &#61; null;private static ReentrantLock lock &#61; new ReentrantLock();private SingletonLazy2(){System.out.println("SingletonLazy2懒汉式构造方法");}/* public static synchronized SingletonLazy2 getInstance(){if (instance&#61;&#61;null){instance &#61; new SingletonLazy2();}return instance;}*//* public static SingletonLazy2 getInstance(){synchronized(SingletonLazy2.class){if (instance&#61;&#61;null){instance &#61; new SingletonLazy2();}return instance;}}*/public static SingletonLazy2 getInstance(){try{lock.lock();if (instance&#61;&#61;null){instance &#61; new SingletonLazy2();}return instance;}finally {lock.unlock();}}
}

解决了线程不安全问题

效率太低了&#xff0c;每个线程在想获得类的实例时候&#xff0c;执行getInstance()方法都要进行同步。而其实这个方法只执行一次实例化代码就够了&#xff0c;后面的想获得该类实例&#xff0c;直接return就行了。方法进行同步效率太低

结论&#xff1a;在实际开发中&#xff0c;不推荐使用这种方式


2.3 双重检查(推荐)

class SingletonLazy4 {//双重检查DCL/*必须加volatile可能发生指令重排问题 概率很小*/// private static SingletonLazy4 instance &#61; null;private static volatile SingletonLazy4 instance &#61; null;private SingletonLazy4(){System.out.println("SingletonLazy3懒汉式构造方法");}public static SingletonLazy4 getInstance(){if (instance&#61;&#61;null){synchronized (SingletonLazy4.class){if (instance&#61;&#61;null){instance &#61; new SingletonLazy4();}}}return instance;}
}
public class SingletonTest {public static void main(String[] args) {for (int i &#61; 0; i <100000; i&#43;&#43;) {new Thread(()->{DoubleDCLSingle doubleDCLSingle &#61; DoubleDCLSingle.getInstance();System.out.println(doubleDCLSingle);},String.valueOf(i)).start();}}
}

Double-Check概念是多线程开发中常使用到的&#xff0c;如代码中所示&#xff0c;我们进行了两次if (singleton &#61;&#61; null)检查&#xff0c;这样就可以保证线程安全了。

这样&#xff0c;实例化代码只用执行一次&#xff0c;后面再次访问时&#xff0c;判断if (singleton &#61;&#61; null)&#xff0c;直接return实例化对象&#xff0c;也避免的反复进行方法同步.

线程安全&#xff1b;延迟加载&#xff1b;效率较高

结论&#xff1a;在实际开发中&#xff0c;推荐使用这种单例设计模式

2.2.4 静态内部类&#xff08;推荐&#xff09;

package com.design.singleton;/*** &#64;author nzy* &#64;create 2021-12-14 16:15* 使用静态内部类 推荐使用* 利用jvm帮助我们保证线程安全性*/
public class StaticInnerSingle {//构造器私有化private StaticInnerSingle() {System.out.println("StaticInnerSingle构造方法");}//写一个静态内部类,该类中有一个静态属性 Singletonprivate static class SingletonInstance {public static final StaticInnerSingle INSTANCE &#61; new StaticInnerSingle();}// 提供一个静态的公有方法&#xff0c;直接返回SingletonInstance.INSTANCEpublic static StaticInnerSingle getInstance() {return SingletonInstance.INSTANCE;}
}
public class SingletonTest {public static void main(String[] args) {for (int i &#61; 0; i <100; i&#43;&#43;) {new Thread(()->{StaticInnerSingle staticInnerSingle &#61; StaticInnerSingle.getInstance();System.out.println(staticInnerSingle);},String.valueOf(i)).start();}}
}

这种方式采用了类装载的机制来保证初始化实例时只有一个线程。

静态内部类方式在Singleton类被装载时并不会立即实例化&#xff0c;而是在需要实例化时&#xff0c;调用getInstance方法&#xff0c;才会装载SingletonInstance类&#xff0c;从而完成Singleton的实例化。

类的静态属性只会在第一次加载类的时候初始化&#xff0c;所以在这里&#xff0c;JVM帮助我们保证了线程的安全性&#xff0c;在类进行初始化时&#xff0c;别的线程是无法进入的。

优点&#xff1a;避免了线程不安全&#xff0c;利用静态内部类特点实现延迟加载&#xff0c;效率高

结论&#xff1a;推荐使用


3总结

按照实例对象被创建的时机&#xff0c;可以将单例模式分为两类。如果在应用开始时创建单例实例&#xff0c;就称作提前加载单例模式&#xff1b;如果在getInstance方法首次被调用时才调用单例构造器&#xff0c;则称作延迟加载单例模式。

单例模式在内存中只有一个实例&#xff0c;减少了内存开支&#xff0c;特别是一个对象需要频繁地创建、销毁时&#xff0c;而且创建或销毁时性能又无法优化&#xff0c;单例模式的优势就非常明显。

当想实例化一个单例类的时候&#xff0c;必须要记住使用相应的获取对象的方法&#xff0c;而不是使用 new。

单例模式是23个模式中比较简单的模式&#xff0c;应用也非常广泛&#xff0c;

如在Spring中&#xff0c;每个Bean默认就是单例的&#xff0c;这样做的优点是Spring容器可以管理这些Bean的生命期&#xff0c;决定什么时候创建出来&#xff0c;什么时候销毁&#xff0c;销毁的时候要如何处理&#xff0c;等等。如果采用非单例模式&#xff08;Prototype类型&#xff09;&#xff0c;则Bean初始化后的管理交由J2EE容器&#xff0c;Spring容器不再跟踪管理Bean的生命周期。

JDK中&#xff0c;java.lang.Runtime 就是经典的饿汉式单例模式


推荐阅读
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 本文介绍了C#中生成随机数的三种方法,并分析了其中存在的问题。首先介绍了使用Random类生成随机数的默认方法,但在高并发情况下可能会出现重复的情况。接着通过循环生成了一系列随机数,进一步突显了这个问题。文章指出,随机数生成在任何编程语言中都是必备的功能,但Random类生成的随机数并不可靠。最后,提出了需要寻找其他可靠的随机数生成方法的建议。 ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • eclipse学习(第三章:ssh中的Hibernate)——11.Hibernate的缓存(2级缓存,get和load)
    本文介绍了eclipse学习中的第三章内容,主要讲解了ssh中的Hibernate的缓存,包括2级缓存和get方法、load方法的区别。文章还涉及了项目实践和相关知识点的讲解。 ... [详细]
  • 本文介绍了Java高并发程序设计中线程安全的概念与synchronized关键字的使用。通过一个计数器的例子,演示了多线程同时对变量进行累加操作时可能出现的问题。最终值会小于预期的原因是因为两个线程同时对变量进行写入时,其中一个线程的结果会覆盖另一个线程的结果。为了解决这个问题,可以使用synchronized关键字来保证线程安全。 ... [详细]
  • 本文详细介绍了Spring的JdbcTemplate的使用方法,包括执行存储过程、存储函数的call()方法,执行任何SQL语句的execute()方法,单个更新和批量更新的update()和batchUpdate()方法,以及单查和列表查询的query()和queryForXXX()方法。提供了经过测试的API供使用。 ... [详细]
  • 标题: ... [详细]
  • 本文介绍了南邮ctf-web的writeup,包括签到题和md5 collision。在CTF比赛和渗透测试中,可以通过查看源代码、代码注释、页面隐藏元素、超链接和HTTP响应头部来寻找flag或提示信息。利用PHP弱类型,可以发现md5('QNKCDZO')='0e830400451993494058024219903391'和md5('240610708')='0e462097431906509019562988736854'。 ... [详细]
  • 本文介绍了Android 7的学习笔记总结,包括最新的移动架构视频、大厂安卓面试真题和项目实战源码讲义。同时还分享了开源的完整内容,并提醒读者在使用FileProvider适配时要注意不同模块的AndroidManfiest.xml中配置的xml文件名必须不同,否则会出现问题。 ... [详细]
  • Java自带的观察者模式及实现方法详解
    本文介绍了Java自带的观察者模式,包括Observer和Observable对象的定义和使用方法。通过添加观察者和设置内部标志位,当被观察者中的事件发生变化时,通知观察者对象并执行相应的操作。实现观察者模式非常简单,只需继承Observable类和实现Observer接口即可。详情请参考Java官方api文档。 ... [详细]
  • Linux服务器密码过期策略、登录次数限制、私钥登录等配置方法
    本文介绍了在Linux服务器上进行密码过期策略、登录次数限制、私钥登录等配置的方法。通过修改配置文件中的参数,可以设置密码的有效期、最小间隔时间、最小长度,并在密码过期前进行提示。同时还介绍了如何进行公钥登录和修改默认账户用户名的操作。详细步骤和注意事项可参考本文内容。 ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • 关键词:Golang, Cookie, 跟踪位置, net/http/cookiejar, package main, golang.org/x/net/publicsuffix, io/ioutil, log, net/http, net/http/cookiejar ... [详细]
author-avatar
冬季梅花1991_156
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有