6、单例模式 Singleton模式 只有一个实例 创建型模式
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模式 只有一个实例 创建型模式的更多相关文章
- ANDROID 中设计模式的採用--创建型模式
所谓模式就是在某一情景下解决某个问题的固定解决方式. 全部的创建型模式都是用作对象的创建或实例化的解决方式. 1 简单工厂模式 创建对象的最简单方法是使用new来创建一个对象,假设仅仅创建一种固 ...
- Java 23种设计模式详尽分析与实例解析之一--创建型模式
面向对象的设计原则 常用的面向对象设计原则包括7个,这些原则并不是独立存在的,它们相互依赖.互为补充. Java设计模式 创建型模式 简单工厂模式 模式动机: 考虑一个简单的软件应用场景,一个软件系统 ...
- 设计模式之美:Creational Patterns(创建型模式)
创建型模式(Creational Patterns)抽象了对象实例化过程. 它们帮助一个系统独立于如何创建.组合和表示它的那些对象. 一个类创建型模式使用继承改变被实例化的类. 一个对象创建型模式将实 ...
- .NET设计模式(7):创建型模式专题总结(Creational Pattern)(转)
概述 创建型模式,就是用来创建对象的模式,抽象了实例化的过程.它帮助一个系统独立于如何创建.组合和表示它的那些对象.本文对五种常用创建型模式进行了比较,通过一个游戏开发场景的例子来说该如何使用创建型模 ...
- .NET设计模式(7):创建型模式专题总结(Creational Pattern)
):创建型模式专题总结(Creational Pattern) 创建型模式专题总结(Creational Pattern) --.NET设计模式系列之七 Terrylee,2006年1月 转载: ...
- NET设计模式 第二部分 创建型模式(6):创建型模式专题总结(Creational Pattern)
创建型模式专题总结(Creational Pattern) ——.NET设计模式系列之七 Terrylee,2006年1月 概述 创建型模式,就是用来创建对象的模式,抽象了实例化的过程.它帮助一个系统 ...
- 设计模式GOF23(创建型模式)
• 创建型模式: 单例模式.工厂模式.抽象工厂模式.建造者模式.原型模式. • 结构型模式: –适配器模式.桥接模式.装饰模式.组合模式.外观模式.享元模式.代理模式. • 行为型模式: 模 ...
- 工厂方法模式——创建型模式02
1. 简单工厂模式 在介绍工厂方法模式之前,先介绍一下简单工厂模式.虽然简单工厂模式不属于GoF 23种设计模式,但通常将它作为学习其他工厂模式的入门,并且在实际开发中使用的也较为频繁. (1 ...
- 设计模式学习之简单工厂(Simple Factory,创建型模式)(1)
简单工厂(Simple Factory,创建型模式) 第一步: 比如我们要采集苹果和香蕉,那么我们需要创建一个Apple类和Banana类,里面各自有采集方法get(),然后通过main方法进行调用, ...
随机推荐
- 数据可视化之powerBI技巧(五)在Power BI中写出优雅的度量值是什么体验?
之前的文章(采悟:连接表的几个DAX函数,一次全掌握)介绍了产品A的客户与产品B的客户的各种交叉关系,其中最常用的应该是找出A和B的共同客户,以便进行产品关联分析. 之前的思路是计算出两个产品的共同客 ...
- Unity-内存
editor 和runtime的内存管理分开的 unity检测不到native内存容量 如c++,lua 一个asset一个ab的问题在于 每个asset都有对应的文件头,并不划算 IL2CPP抛弃了 ...
- freeMarker随手笔记
freemarker官网:http://docs.freemarker.cn/ 注意: 1.如果标签没有嵌套内容(在开始标签和结束标签之间的内容),那么可以只使用开始标签 (详情:http://fre ...
- PHP 反序列化漏洞入门学习笔记
参考文章: PHP反序列化漏洞入门 easy_serialize_php wp 实战经验丨PHP反序列化漏洞总结 PHP Session 序列化及反序列化处理器设置使用不当带来的安全隐患 利用 pha ...
- elementui 使用Form表单 的 resetForm表单功能出现的问题
代码因为在保密机上,这里只进行描述并截取elemen文档中的代码作为参考 今天在开发一个很简单需求的时候遇到的问题,在使用elementui的表单功能,将增和改的表单进行了复用,是在表单的父组件 dr ...
- Ethical Hacking - Web Penetration Testing(10)
SQL INJECTION SQLMAP Tool designed to exploit SQL injections. Works with many DB types, MySQL, MSSQL ...
- CUDA中关于C++特性的限制
CUDA中关于C++特性的限制 CUDA官方文档中对C++语言的支持和限制,懒得每次看英文文档,自己尝试翻译一下(没有放lambda表达式的相关内容,太过于复杂,我选择不用).官方文档https:// ...
- swfupload控件文件上传大小限制设置
swfupload控件,是我在开发过程中用到的上传文件的控件,非常实用和方便.但最近碰到一些问题,解决之后进行一下整理. 因为用户上传文件的大小限制增加,导致原本上传控件时,文件的大小需要进行调整和限 ...
- consul++ansible+shell批量下发注册node_exporter
--日期:2020年7月21日 --作者:飞翔的小胖猪 文档功能说明: 文档通过ansible+shell+consul的方式实现批量下发安装Linux操作系统监控的node_exporter软件, ...
- java计算下一个整5分钟时间点
需求背景 我的需求是获取当前时间之后的下一个"整5分钟时间点". 首先,那么何为"整5分钟时间点"? 满足以下两个条件的时间: 分钟数等于以下时间中的一个,且秒 ...