23种设计模式之创建型模式 - 单例模式

文章目录

  • 一、单例模式
    • 1.1单例模式定义
    • 1.2 单例模式的特点
  • 二、实现单例模式的方式
    • 2.1 饿汉式
    • 2.2 懒汉式
    • 2.3 双重检查锁:
    • 2.4 静态内部类
    • 2.5 枚举实现(防止反射攻击):

一、单例模式

1.1单例模式定义

单例模式确保系统中某个类只有一个实例,并提供一个访问它的全局访问点。主要解决一个全局使用的类频繁地创建与销毁,控制实例数目,节省系统资源。

1.2 单例模式的特点

  • 单例类只能有一个实例
  • 单例类必须自己创建自己的唯一实例
  • 单例类必须给所有其他对象提供这一实例
  • 单例模式保证了全局对象的唯一性,比如系统启动读取配置文件就需要单例保证配置的一致性

单例的四大原则:

  • 构造私有
  • 以静态方法或者枚举返回实例
  • 确保实例只有一个,尤其是多线程环境
  • 确保反序列换时不会重新构建对象

二、实现单例模式的方式

2.1 饿汉式

饿汉式单例在类加载初始化时就创建好一个静态的对象供外部使用,除非系统重启,这个对象不会改变,所以本身就是线程安全的。
Singleton 通过将构造方法限定为 private 避免了类在外部被实例化,在同一个虚拟机范围内,Singleton 的唯一实例只能通过 getInstance()方法访问。(事实上,通过 Java 反射机制是能够实例化构造方法为 private 的类的,会使 Java单例实现失效)

/*** @Author huang.bX* @Date 2021/7/21*/
public class SingletonTest01 {public static void main(String[] args) {Hungry instance = Hungry.getInstance();Hungry instance1 = Hungry.getInstance();Hungry instance2 = Hungry.getInstance();System.out.println(instance.getClass());System.out.println(instance1.getClass());System.out.println(instance2.getClass());}
}//饿汉式
class Hungry {//1构造器私有化,外部不能直接newprivate Hungry() {}//2本类的内部创建实例private final static Hungry hungry = new Hungry();//提供一个全局访问点共有的静态方法 返回实例对象public static Hungry getInstance(){return hungry;}
}

在这里插入图片描述

2.2 懒汉式

该示例虽然用延迟加载方式实现了懒汉式单例,但在多线程环境下会产生多个 Singleton 对象;

/*** @Author huang.bX* @Date 2021/7/21*/
public class SingletonTest03 implements Runnable {@Overridepublic void run(){for (int i=1;i<1000;i++){LazyMan lazyMan=LazyMan.getInstance();System.out.println(lazyMan.hashCode());}}public static void main(String[] args) {/**   LazyMan instance1 = LazyMan.getInstance();*   LazyMan instance2 = LazyMan.getInstance();*  System.out.println(instance1.hashCode()==instance2.hashCode());*/new Thread(new SingletonTest03()).start();new Thread(new SingletonTest03()).start();new Thread(new SingletonTest03()).start();}
}class LazyMan{private LazyMan(){}private static LazyMan lazyMan;//public static LazyMan getInstance()线程不安全public static synchronized LazyMan getInstance(){if (lazyMan==null){lazyMan = new LazyMan();}return lazyMan;}
}

在这里插入图片描述

2.3 双重检查锁:

使用双重检查锁进一步做了优化,可以避免整个方法被锁,只对需要锁的代码部分加锁,可以提高执行效率。

/*** @Author huang.bX* @Date 2021/7/21*/
public class SingletonTest05 {public static void main(String[] args) {DoubleLock instance1 = DoubleLock.getInstance();DoubleLock instance2 = DoubleLock.getInstance();System.out.println(instance1.hashCode()==instance2.hashCode());}
}class DoubleLock{private static volatile DoubleLock doubleLock;private DoubleLock(){}public static DoubleLock getInstance(){if (doubleLock==null){synchronized (DoubleLock.class){if (doubleLock==null){doubleLock = new DoubleLock();}}}return doubleLock;}
}

在这里插入图片描述

2.4 静态内部类

这种方式引入了一个内部静态类(static class),静态内部类只有在调用时才会加载,它保证了 Singleton 实例的延迟初始化,又保证了实例的唯一性。它把 singleton 的实例化操作放到一个静态内部类中,在第一次调用 getInstance() 方法时,JVM 才会去加载 InnerObject 类,同时初始hsingleton 实例,所以能让 getInstance() 方法线程安全。特点是:即能延迟加载,也能保证线程安全。静态内部类虽然保证了单例在多线程并发下的线程安全性,但是在遇到序列化对象时,默认的方式运行得到的结果就是多例的。

/*** @Author huang.bX* @Date 2021/7/21*/
public class SingletonTest06 {public static void main(String[] args) {StaticInClass instance1 = StaticInClass.getInstance();StaticInClass instance2= StaticInClass.getInstance();System.out.println(instance1.hashCode()==instance2.hashCode());}
}class StaticInClass{private static volatile StaticInClass staticInClass;//构造器私有化private StaticInClass(){}//定义一个静态内部类,该类中有一个静态属性private static class Inner{private static final StaticInClass INSTANCE = new StaticInClass();}public static synchronized StaticInClass getInstance(){return Inner.INSTANCE;}}

在这里插入图片描述

2.5 枚举实现(防止反射攻击):

事实上,通过 Java 反射机制是能够实例化构造方法为 private 的类的。这也就是我们现在需要引入的枚举单例模式。

/*** @Author huang.bX* @Date 2021/7/21*/
public class SingletonTest07 {public static void main(String[] args) {Singleton instance1 = Singleton.INSTANCE;Singleton instance2 = Singleton.INSTANCE;System.out.println(instance1.hashCode());System.out.println(instance2.hashCode());System.out.println(instance2.hashCode()==instance1.hashCode());System.out.println(instance1.getClass());System.out.println(instance2.getClass());System.out.println(instance1.getDeclaringClass());}
}enum Singleton{INSTANCE;//属性public void say(){System.out.println("ok!");}
}

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/572661.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

表格中的状态类型值(tag)

一&#xff1a;数字转换为简单的中文值 ** 不用转换直接用find()方法&#xff1a;在statusList里找&#xff1b; **lastHandleCode是对应的获取到的每行数据的code值&#xff1b; vue: <el-table-column label"执行状态" align"center"><templat…

Collection与数据结构 链表与LinkedList (一):链表概述与单向无头非循环链表实现

1.ArrayList的缺点 上篇文章我们已经对顺序表进行了实现,并且对ArrayList进行了使用,我们知道ArrayList底层是使用数组实现的. 由于其底层是一段连续空间&#xff0c;当在ArrayList任意位置插入或者删除元素时&#xff0c;就需要将后序元素整体往前或者往后搬移&#xff0c;时…

SpringMvc之映射器HandlerMapping

简介 在springmvc的处理流程中&#xff0c;第一步就是查询请求对应的映射器&#xff0c;然后组装成处理器链处理请求&#xff0c;本文意在梳理该过程 重要实现 HandlerMapping是一个接口&#xff0c;该接口用于通过HttpServletRequest寻找对应的处理器&#xff0c;接口介绍如下…

windows系统安装RabbitMQ

RabbitMQ是实现了高级消息队列协议&#xff08;AMQP&#xff09;的开源消息代理软件&#xff08;亦称面向消息的中间件&#xff09;。RabbitMQ服务器是用Erlang语言编写的&#xff0c;而集群和故障转移是构建在开放电信平台框架上的。所有主要的编程语言均有与代理接口通讯的客…

PL/SQL概述

oracle从入门到总裁:​​​​​​https://blog.csdn.net/weixin_67859959/article/details/135209645 PL/SQL概述 PL/SQL(Procedural Language extension to SQL)是 Oracle 对标准 SQL语言的扩充&#xff0c;是专门用于各种环境下对 Oracle 数据库进行访问和开发的语言。 由…

【有限状态机】- FSM详细讲解 【附Autoware有限状态机模型代码讲解】

参考博客&#xff1a; &#xff08;1&#xff09;FSM&#xff08;有限状态机&#xff09; &#xff08;2&#xff09;关于有限状态机(FSM)的一些思考 &#xff08;3&#xff09;状态设计模式 1 状态机简介 有限状态机FSM&#xff1a;有限个状态以及在这些状态之间的转移和动作…

文献阅读工具-->Adobe pdf + 有道词典

Adobe pdf 有道词典 最近一直在考虑用什么文献阅读工具&#xff0c;痛点无非就是想用翻译功能&#xff0c;Adobe pdf的添加注释已经很好用了&#xff0c;使用了zotero&#xff0c;感觉不行&#xff08;不能直接对原文件修改&#xff0c;有副本&#xff0c;麻烦&#xff09;。…

帆软报表在arm架构的linux

有朋友遇到一个问题在部署帆软报表时遇到报错。 问 我在 arm架构的linux服务器上部署帆软报表遇到了一个棘手的问题&#xff0c;你有空帮忙看下嘛。 我看后台日志报的错是 需要升级 gcc、libmawt.so &#xff0c;是系统中缺少Tomcat需要的依赖库&#xff0c;你之前处理过类似…

服务器监控软件夜莺采集监控(三)

文章目录 一、采集器插件1. exec插件2. rabbitmq插件3. elasticsearch插件 二、监控仪表盘1. 系统信息2. 数据服务3. NginxMQ4. Docker5. 业务日志 一、采集器插件 1. exec插件 input.exec/exec.toml [[instances]] commands ["/home/monitor/categraf/scripts/*.sh&q…

6.二叉树——2.重建树

已知先序和中序序列 根据先序序列找到树根根据树根和中序序列找到左右子树 同理根据后序序列和中序序列也能重构树&#xff0c;但前序和后序不可以 递归coding思路 设先序序列为preorder[n]&#xff0c;中序序列为midorder[n] 大事化小&#xff1a; 确定根&#xff0c;即树…

uniapp写小程序如何实现分包

众所众知小程序上传的过程中对包的大小有限制&#xff0c;正常情况下不允许当个包超过2M&#xff0c;所以需要分包 需要再pages.json这个文件夹中进行配置 "pages": [{"path": "pages/index/index","style": {"navigationBarTit…

基于java实现学科竞赛管理系统【Springboot+mybatis+layui】

基于java实现学科竞赛管理系统【Springbootmybatislayui】 博主介绍&#xff1a;多年java开发经验&#xff0c;专注Java开发、定制、远程、文档编写指导等,csdn特邀作者、专注于Java技术领域 作者主页 央顺技术团队 Java毕设项目精品实战案例《1000套》 欢迎点赞 收藏 ⭐留言 文…