一、什么是单例模式:

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

二、单例模式的特点:

  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. djiango的模板语言(template)

    老师的博客:http://www.cnblogs.com/liwenzhou/p/7931828.html 官方文档:https://docs.djangoproject.com/en/1.11/re ...

  2. jdk源码阅读笔记-Integer

    public final class Integer extends Number implements Comparable<Integer> Integer 由final修饰了,所以该 ...

  3. 对Tomcat 8.0进行JVM层面的优化(基于Oracle JDK 8)

    目录 1 Tomcat的内存调优 1.1 Tomcat的内存占用 1.2 内存配置相关参数 1.3 内存调优实践 1.4 验证配置效果 2 GC策略调优实践 Tomcat容器是运行在JVM上的, 其默 ...

  4. mysql实现主从备份

    mysql 主从备份的原理: 主服务器在做数据库操作的时候将所有的操作通过日志记录在binlog里面,有专门的文件存放.如localhost-bin.000003,这种,从服务器 和主服务配置好关系后 ...

  5. 【效率神奇】Github丧心病狂的9个狠招

    Github,一个被业内朋友成为「全球最大的同性交友社区」的平台. 小时候遇到不会的字可以查新华字典.后来写作文我们可以通过作文书.或者文摘去找合适的素材.同样,写代码可以去Github上找适合自己的 ...

  6. 使用工厂模式解耦和IoC思想

    使用工厂模式解耦. 一.需求场景: 某一层功能需要改动,但其他层代码不变 实现类1:MyDaoImpl查询自己的数据库. ====改为====> 实现类2:MyDaoImpl2从其它地址得到数据 ...

  7. 前端基础之--css中可被继承和不可被继承的属性

    一.无继承性的属性 1.display:规定元素应该生成的框的类型 2.文本属性:vertical-align:垂直文本对齐 text-decoration:规定添加到文本的装饰 text-shado ...

  8. cesium 之地图切换展示效果篇(附源码下载)

    前言 cesium 官网的api文档介绍地址cesium官网api,里面详细的介绍 cesium 各个类的介绍,还有就是在线例子:cesium 官网在线例子,这个也是学习 cesium 的好素材. 内 ...

  9. andorid 应用第二次登录实现自动登录

    前置条件是所有用户相关接口都走 https,非用户相关列表类数据走 http. 步骤 第一次登陆 getUserInfo 里带有一个长效 token,该长效 token 用来判断用户是否登陆和换取短 ...

  10. 【Oracle RAC】Linux系统Oracle11gR2 RAC安装配置详细过程V3.1(图文并茂)

    [Oracle RAC]Linux系统Oracle11gR2 RAC安装配置详细过程V3.1(图文并茂) 2 Oracle11gR2 RAC数据库安装准备工作2.1 安装环境介绍2.2 数据库安装软件 ...