Java 单例(Singleton)模式
一、什么是单例模式:
单例模式是一种确保了一个类只有一个实例,而且自行实例化并向整个系统提供这个实例。被实例化的类称为单例类。
二、单例模式的特点:
- 单例类只有一个实例。
- 单例类必须自行创建自己唯一的实例。
- 单例类必须给其他对象提供这个实例。
注意:虽然单例模式和单例类限定了只能有一个实例,但是作为单例模式的推广,可以推广到任意且有限多个实例的情况,这时候被称为多例模式和多例类。
三、单例模式的结构:

- 一个单例类只有一个实例。
- 单例类的实例持有对自己的引用。
四、单例模式的实例化:
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,可能会给其他开发人员造成困扰,特别是看不到源码的时候。
懒汉式单例类与饿汉式单例类的比较:
- 饿汉式单例类在自己被加载时将自己实例化,即便加载器是静态的,依旧在加载时实例化自己;懒汉式单例类在第一次被引用时将自己实例化,如果加载器是静态的,懒汉式单例类被加载时不会将自己实例化。
- 从资源利用角度讲,懒汉式单例类的资源利用效率高点;从速度和反应时间来讲,饿汉式的好点。
Java 单例(Singleton)模式的更多相关文章
- 【Java学习笔记之三十】详解Java单例(Singleton)模式
概念: Java中单例模式是一种常见的设计模式,单例模式的写法有好几种,这里主要介绍三种:懒汉式单例.饿汉式单例.登记式单例. 单例模式有以下特点: 1.单例类只能有一个实例. 2.单例类必须自己创建 ...
- 漫谈设计模式(二):单例(Singleton)模式
1.前言 实际业务中,大多业务类只需要一个对象就能完成所有工作,另外再创建其他对象就显得浪费内存空间了,例如web开发中的servlet,这时便要用到单例模式,就如其名一样,此模式使某个类只能生成唯一 ...
- JAVA中实现单例(Singleton)模式的八种方式
单例模式 单例模式,是一种常用的软件设计模式.在它的核心结构中只包含一个被称为单例的特殊类.通过单例模式可以保证系统中,应用该模式的类一个类只有一个实例.即一个类只有一个对象实例. 基本的实现思路 单 ...
- Android与设计模式——单例(Singleton)模式
概念: java中单例模式是一种常见的设计模式.单例模式分三种:懒汉式单例.饿汉式单例.登记式单例三种. 单例模式有一下特点: 1.单例类仅仅能有一个实例. 2.单例类必须自己自己创建自己的唯一实例. ...
- 设计一个线程安全的单例(Singleton)模式
在设计单例模式的时候.尽管非常easy设计出符合单例模式原则的类类型,可是考虑到垃圾回收机制以及线程安全性.须要我们思考的很多其它.有些设计尽管能够勉强满足项目要求,可是在进行多线程设计的时候.不考虑 ...
- 单例Singleton模式的两种实现方法
在设计模式中,有一种叫Singleton模式的,用它可以实现一次只运行一个实例.就是说在程序运行期间,某个类只能有一个实例在运行.这种模式用途比较广泛,会经常用到,下面是Singleton模式的两种实 ...
- 设计模式C++描述----01.单例(Singleton)模式
一.概念 单例模式:其意图是保证一个类仅有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享. class CSingleton { //公有的静态方法,来获取该实例 public: s ...
- java双重检测或枚举类实现线程安全单例(懒汉模式)
双重检测实现 /** * 懒汉模式->双重同步锁单例模式 */ public class SingletonExample5 { private SingletonExample5() { } ...
- 单例/单体模式(Singleton)
单例/单体模式(Singleton) 首先,单例模式是对象的创建模式之一,此外还包括工厂模式. 单例模式的三个特点: 1,该类只有一个实例 2,该类自行创建该实例(在该类内部创建自身的实例对象) 3, ...
- OpenJDK源码研究笔记(十三):Javac编译过程中的上下文容器(Context)、单例(Singleton)和延迟创建(LazyCreation)3种模式
在阅读Javac源码的过程中,发现一个上下文对象Context. 这个对象用来确保一次编译过程中的用到的类都只有一个实例,即实现我们经常提到的"单例模式". 今天,特意对这个上下文 ...
随机推荐
- djiango的模板语言(template)
老师的博客:http://www.cnblogs.com/liwenzhou/p/7931828.html 官方文档:https://docs.djangoproject.com/en/1.11/re ...
- jdk源码阅读笔记-Integer
public final class Integer extends Number implements Comparable<Integer> Integer 由final修饰了,所以该 ...
- 对Tomcat 8.0进行JVM层面的优化(基于Oracle JDK 8)
目录 1 Tomcat的内存调优 1.1 Tomcat的内存占用 1.2 内存配置相关参数 1.3 内存调优实践 1.4 验证配置效果 2 GC策略调优实践 Tomcat容器是运行在JVM上的, 其默 ...
- mysql实现主从备份
mysql 主从备份的原理: 主服务器在做数据库操作的时候将所有的操作通过日志记录在binlog里面,有专门的文件存放.如localhost-bin.000003,这种,从服务器 和主服务配置好关系后 ...
- 【效率神奇】Github丧心病狂的9个狠招
Github,一个被业内朋友成为「全球最大的同性交友社区」的平台. 小时候遇到不会的字可以查新华字典.后来写作文我们可以通过作文书.或者文摘去找合适的素材.同样,写代码可以去Github上找适合自己的 ...
- 使用工厂模式解耦和IoC思想
使用工厂模式解耦. 一.需求场景: 某一层功能需要改动,但其他层代码不变 实现类1:MyDaoImpl查询自己的数据库. ====改为====> 实现类2:MyDaoImpl2从其它地址得到数据 ...
- 前端基础之--css中可被继承和不可被继承的属性
一.无继承性的属性 1.display:规定元素应该生成的框的类型 2.文本属性:vertical-align:垂直文本对齐 text-decoration:规定添加到文本的装饰 text-shado ...
- cesium 之地图切换展示效果篇(附源码下载)
前言 cesium 官网的api文档介绍地址cesium官网api,里面详细的介绍 cesium 各个类的介绍,还有就是在线例子:cesium 官网在线例子,这个也是学习 cesium 的好素材. 内 ...
- andorid 应用第二次登录实现自动登录
前置条件是所有用户相关接口都走 https,非用户相关列表类数据走 http. 步骤 第一次登陆 getUserInfo 里带有一个长效 token,该长效 token 用来判断用户是否登陆和换取短 ...
- 【Oracle RAC】Linux系统Oracle11gR2 RAC安装配置详细过程V3.1(图文并茂)
[Oracle RAC]Linux系统Oracle11gR2 RAC安装配置详细过程V3.1(图文并茂) 2 Oracle11gR2 RAC数据库安装准备工作2.1 安装环境介绍2.2 数据库安装软件 ...