一、什么是单例模式:

单例模式是一种确保了一个类只有一个实例,而且自行实例化并向整个系统提供这个实例。被实例化的类称为单例类。

二、单例模式的特点:

  1. 单例类只有一个实例。
  2. 单例类必须自行创建自己唯一的实例。
  3. 单例类必须给其他对象提供这个实例。

注意:虽然单例模式和单例类限定了只能有一个实例,但是作为单例模式的推广,可以推广到任意且有限多个实例的情况,这时候被称为多例模式和多例类。

三、单例模式的结构:

  1. 一个单例类只有一个实例。
  2. 单例类的实例持有对自己的引用。

四、单例模式的实例化:

Java中单例模式有着自己的特点,具体表现在单例类的实例化上:

饿汉式单例类(静态常量):

 /**
* 饿汉式(静态常量)
*
* @author ZhouDX
* @since 2019/3/4 22:12:28
*/
public class HungerSingleton {
private static final HungerSingleton SINGLETON= new HungerSingleton(); /**
* 私有的默认构造函数
*/
private HungerSingleton() {
} /**
* 静态工厂方法
*/
public static HungerSingleton getInstance() {
return SINGLETON;
}
}

Java中最简单的单例类,类的单例被声明为静态变量,在类加载时,调用类的私有构造函数,静态变量被实例化。

特点:

  1.类的构造函数私有,避免了外界利用构造函数创建任意多的实例。

  2.且由于构造函数私有,类不能被继承。

  3.只能通过静态方法getInstance()来获取类的实例对象。

优点:

  类装载的时候就完成实例化。避免了线程同步问题。

缺点:

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

  饿汉式单例类(静态代码块):

 /**
* 饿汉式单例类(静态代码块)
*
* @author ZhouDX
* @since 2019/3/13 22:45:24
*/
public class HungerSington_StaticCode {
private static HungerSington_StaticCode singleton; /**
* 静态代码块
*/
static {
singleton= new HungerSington_StaticCode();
} /**
* 私有构造函数
*/
private HungerSington_StaticCode() {
} /**
* 获取单例类实例的唯一接口
*
* @return 单例类
*/
public static HungerSington_StaticCode getInstance() {
return singleton;
}
}

  特点:

  将单例类放在静态代码块中,也是类在加载时执行静态代码块中的代码,完成类的实例化,优缺点同静态常量。

汉懒式单例类(线程不安全):

 /**
* 懒汉式(线程不安全)[不可用]
*
* @author ZhouDX
* @since 2019/3/13 22:52:24
*/
public class LazySingleton_ThreadUnsafe {
private static LazySingleton_ThreadUnsafe singleton; /**
* 获取单例类实例
*
* @return 单例类实例
*/
public static LazySingleton_ThreadUnsafe getInstance() {
if (null == singleton) {
singleton = new LazySingleton_ThreadUnsafe();
} return singleton;
}
}

  特点:

  1.达到了延迟加载的目的,只有在单例类第一次被引用时将自己实例化。

  2.在单线程下使用。

  缺点:  

  在多线程的环境中,多个线程同时进入if (null == singleton) {},还未执行singleton = new LazySingleton_ThreadUnsafe()时,另一个线程也恰好进入这里,就会造成单例类多个实例,线程不安全,不可以再多线程的环境下使用。

  懒汉式(线程安全,同步方法)

 /**
* 懒汉式(线程安全,同步方法)
*
* @author ZhouDX
* @since 2019/3/4 22:23:02
*/
public class LazySingleton {
private static LazySingleton lazySingleton = null; /**
* 构造函数
*/
private LazySingleton() {
} /**
* 静态工厂方法,返回懒汉式实力类的唯一实例
*
* @return
*/
public static synchronized LazySingleton getInstance() {
if (lazySingleton == null) {
return lazySingleton = new LazySingleton();
}
return lazySingleton;
}
}

  特点:

  使用了synchronized对静态工厂类方法进行了同步,安全处理多线程的问题。

  缺点:

  每个线程执行getInstance()方法都要进行同步,大大降低了执行的效率。且getInstance()只需要实例化一次就可以。

  懒汉式(线程不安全,同步代码块)

 /**
* 懒汉式(线程安全,同步代码块)
*
* @author ZhouDX
* @since 2019/3/13 23:12:08
*/
public class LaSingleton_ThreadUnsafe {
private static LaSingleton_ThreadUnsafe singleton; /**
* 静态构造方法
*/
private LaSingleton_ThreadUnsafe() {
} /**
* 获取单例类实例
*
* @return 单例类实例
*/
public static LaSingleton_ThreadUnsafe getInstance() {
if (null == singleton) {
synchronized (LaSingleton_ThreadUnsafe.class) {
singleton = new LaSingleton_ThreadUnsafe();
}
}
return singleton;
}
}

  特点:

  同步产生实例的代码块。

  缺点:

  假如一个线程进入了if (singleton == null)判断语句块,还未来得及往下执行,另一个线程也通过了这个判断语句,这时便会产生多个实例。

  懒汉式(双重检查):

 /**
* 懒汉式(双重检查):
*
* @author ZhouDX
* @since 2019/3/13 23:24:27
*/
public class LazySingleton_DoubleCheck {
private static volatile LazySingleton_DoubleCheck singleton; /**
* 静态构造方法
*/
private LazySingleton_DoubleCheck() {
} /**
* 获取单例类实例
*
* @return 单例类实例
*/
public static LazySingleton_DoubleCheck getInstance() {
if (null == singleton) {
synchronized (LazySingleton_DoubleCheck.class) {
singleton = new LazySingleton_DoubleCheck();
}
}
return singleton;
}
}

  特点:

  1.进行了两次if (singleton == null)检查,保证了线程安全。

  2.实例化代码只执行一次,后面再次访问时,判断if (singleton == null),直接return实例化对象。

  优点:

  程安全;延迟加载;效率较高。

  静态内部类:

 /**
* 静态内部类
*
* @author ZhouDX
* @since 2019/3/13 23:28:58
*/
public class Singleton_StaticInnerClass {
/**
* 私有构造方法
*/
private Singleton_StaticInnerClass() {
} /**
* 静态内部类
*/
private static class SingletonInstance {
private static final Singleton_StaticInnerClass SINGLETON = new Singleton_StaticInnerClass();
} /**
* 获取单例类实例
*
* @return 单例类实例
*/
private static Singleton_StaticInnerClass getInstance() {
return SingletonInstance.SINGLETON;
}
}

  特点:

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

  优点:

  避免了线程不安全,延迟加载,效率高。

  枚举:

 /**
* 枚举
*
* @author ZhouDX
* @since 2019/3/13 23:33:43
*/
public enum Singleton_Enum {
SINGLETON; public void whateverMethod() {
}
}

  特点:

  系统内存中该类只存在一个对象,节省了系统资源,对于一些需要频繁创建销毁的对象,使用单例模式可以提高系统性能。

  优点:

  当想实例化一个单例类的时候,必须要记住使用相应的获取对象的方法,而不是使用new,可能会给其他开发人员造成困扰,特别是看不到源码的时候。

懒汉式单例类与饿汉式单例类的比较:

  1. 饿汉式单例类在自己被加载时将自己实例化,即便加载器是静态的,依旧在加载时实例化自己;懒汉式单例类在第一次被引用时将自己实例化,如果加载器是静态的,懒汉式单例类被加载时不会将自己实例化。
  2. 从资源利用角度讲,懒汉式单例类的资源利用效率高点;从速度和反应时间来讲,饿汉式的好点。

Java 单例(Singleton)模式的更多相关文章

  1. 【Java学习笔记之三十】详解Java单例(Singleton)模式

    概念: Java中单例模式是一种常见的设计模式,单例模式的写法有好几种,这里主要介绍三种:懒汉式单例.饿汉式单例.登记式单例. 单例模式有以下特点: 1.单例类只能有一个实例. 2.单例类必须自己创建 ...

  2. 漫谈设计模式(二):单例(Singleton)模式

    1.前言 实际业务中,大多业务类只需要一个对象就能完成所有工作,另外再创建其他对象就显得浪费内存空间了,例如web开发中的servlet,这时便要用到单例模式,就如其名一样,此模式使某个类只能生成唯一 ...

  3. JAVA中实现单例(Singleton)模式的八种方式

    单例模式 单例模式,是一种常用的软件设计模式.在它的核心结构中只包含一个被称为单例的特殊类.通过单例模式可以保证系统中,应用该模式的类一个类只有一个实例.即一个类只有一个对象实例. 基本的实现思路 单 ...

  4. Android与设计模式——单例(Singleton)模式

    概念: java中单例模式是一种常见的设计模式.单例模式分三种:懒汉式单例.饿汉式单例.登记式单例三种. 单例模式有一下特点: 1.单例类仅仅能有一个实例. 2.单例类必须自己自己创建自己的唯一实例. ...

  5. 设计一个线程安全的单例(Singleton)模式

    在设计单例模式的时候.尽管非常easy设计出符合单例模式原则的类类型,可是考虑到垃圾回收机制以及线程安全性.须要我们思考的很多其它.有些设计尽管能够勉强满足项目要求,可是在进行多线程设计的时候.不考虑 ...

  6. 单例Singleton模式的两种实现方法

    在设计模式中,有一种叫Singleton模式的,用它可以实现一次只运行一个实例.就是说在程序运行期间,某个类只能有一个实例在运行.这种模式用途比较广泛,会经常用到,下面是Singleton模式的两种实 ...

  7. 设计模式C++描述----01.单例(Singleton)模式

    一.概念 单例模式:其意图是保证一个类仅有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享. class CSingleton { //公有的静态方法,来获取该实例 public: s ...

  8. java双重检测或枚举类实现线程安全单例(懒汉模式)

    双重检测实现 /** * 懒汉模式->双重同步锁单例模式 */ public class SingletonExample5 { private SingletonExample5() { } ...

  9. 单例/单体模式(Singleton)

    单例/单体模式(Singleton) 首先,单例模式是对象的创建模式之一,此外还包括工厂模式. 单例模式的三个特点: 1,该类只有一个实例 2,该类自行创建该实例(在该类内部创建自身的实例对象) 3, ...

  10. OpenJDK源码研究笔记(十三):Javac编译过程中的上下文容器(Context)、单例(Singleton)和延迟创建(LazyCreation)3种模式

    在阅读Javac源码的过程中,发现一个上下文对象Context. 这个对象用来确保一次编译过程中的用到的类都只有一个实例,即实现我们经常提到的"单例模式". 今天,特意对这个上下文 ...

随机推荐

  1. [Swift]LeetCode1033. 移动石子直到连续 | Moving Stones Until Consecutive

    Three stones are on a number line at positions a, b, and c. Each turn, let's say the stones are curr ...

  2. 从壹开始前后端 [vue后台] 之一 || 权限后台系统 1.0 正式上线

    缘起 哈喽各位小伙伴周三好,春节已经过去好多天了,群里小伙伴也各种催搞了,新年也接了新项目,比较忙,不过还是终于赶上这个二月的尾巴写了这篇文章,也把 vue 权限后台上线了(项目地址:http://1 ...

  3. Python 转化成 PB 格式数据

    一.概述 Protocol Buffers 是 Google 公司开发的一种轻便高效的结构化数据存储格式,可以用于结构化数据串行化,或者说序列化.它很适合做数据存储或 RPC 数据交换格式.可用于通讯 ...

  4. DSAPI 生成桌面图标(带数字)

    功能:在桌面上创建一个带有指定数字的图标. 效果图: 生成的ICO图标 代码 Private Sub 生成桌面图标(消息数量 As Integer) Try Dim B As New Bitmap(M ...

  5. C# 数组比较--取得两个集合的交集,差集,并集的方法

    方法关键字: 交集:Intersect 差集:Except 并集:Union 使用代码: , , , , }; , , , , }; var 交集 = arr1.Intersect(arr2).ToL ...

  6. WordPress怎样设置菜单栏旋转小图标

    最近我在浏览别的博客的文章时,无意间发现了一个很好看的小装饰.那就是在WordPress菜单栏上的小图标.于是我研究了研究,弄到了设置方法之后决定把它分享出来. 菜单栏的小图标 设置步骤: 1, 我们 ...

  7. 只用最适合的!全面对比主流 .NET 报表控件

    本文由葡萄城技术团队于博客园原创并首发 转载请注明出处:葡萄城官网,葡萄城为开发者提供专业的开发工具.解决方案和服务,赋能开发者. 前言 随着 .NET 平台的出现,报表相关的开发控件随着而来,已经有 ...

  8. 章节十、2-用Linktext和PartialLinkText、ClassName、TagName定位元素

    一.通过内容的方式定位元素 使用Linktext和PartialLinkText定位元素的前提需要"文本"在“a”标签内,selenium才可以找到链接文本或者部分链接文本的元素. ...

  9. github 进阶说明

    目录 github 进阶说明 前言 三个目录树 重置 git reset 增加路径的reset 检出 checkout 带路径的checkout 仓库 数据对象 其他 资料 github 进阶说明 前 ...

  10. windows下nginx的安装及使用

    安装过程比较简单 1.下载nginx http://nginx.org/en/download.html 下载稳定版本,以nginx/Windows-1.14.2为例,直接下载 nginx-1.14. ...