Java单例模式几种实现方式
在平时的工作、学员的学习以及面试过程中,单例模式作为一种常用的设计模式,会经常被面试官问到,甚至笔试会要求学员现场默写,下面将会就单例模式的实现思路和几种常见的实现方式进行简单的分享。
单例模式,是一种常用的软件设计模式。在它的核心结构中只包含一个被称为单例的特殊类。通过单例模式可以保证系统中,应用该模式的类一个类只有一个实例。即一个类只有一个对象实例。是最常用到的设计模式之一,熟悉设计模式的朋友对单例模式都不会陌生。一般介绍单例模式的书籍都会提到 饿汉式 和 懒汉式 这两种实现方式。但是除了这两种方式,本文还会介绍其他几种实现单例的方式。
基本的实现思路
单例模式要求类能够有返回对象一个引用(永远是同一个)和一个获得该实例的方法(必须是静态方法,通常使用getInstance这个名称)。
通俗的讲:即设计一个类,在整个应用中只存在一个对象(需要做到让本类的外部不能够随意创建对象)
单例的实现主要是通过以下三个步骤:
① 构造方法私有化(暂时不考虑反射)
② 在内部创建好一个对象并保存起来
③ 向外提供一个公共的静态的方法,返回内部对象的地址
第一种实现方式:饿汉式[静态常量]的方式
示例代码:
package cn.itsource.sington;
/**
* 饿汉式[静态常量]
*
* @author Administrator
*
*/
public class SingletonTest1 {
/**
* 单例模式 设计一个类 ,让这个 类中只存在一个对象的实例
*
* 1 需要一个私有的 构造方法,这样可以避免外部创建对象
*/
private SingletonTest1() {
}
/*
* 2 在本类内部创建一个 对象 ,保存起来
*/
private static SingletonTest1 instance = new SingletonTest1();
/*
* 3 向外公布一个 公共的 静态的方法 ,返回内部保存的这个对象
*
*/
public static SingletonTest1 getInstance() {
return instance;
}
}
优点:这种写法比较简单,就是在类装载的时候就完成实例化。避免了线程同步问题。
缺点:对象的创建是类加载的时候,可能会导致类加载很慢,没有达到Lazy Loading的效果。如果从始至终从未使用过这个实例,则会造成内存的浪费。
第二种实现方式:饿汉式[静态代码块]的方式
示例代码:
package cn.itsource.sington;
public class SingletonTest3 {
private SingletonTest3() {
}
private static SingletonTest3 instance;
static {
instance = new SingletonTest3();
}
public static SingletonTest3 getInstance() {
return instance;
}
}
这种方式和上面的方式其实类似,只不过将类实例化的过程放在了静态代码块中,也是在类装载的时候,就执行静态代码块中的代码,初始化类的实例。优缺点和上面是一样的。
第三种实现方式:懒汉式[双重检查]的方式
示例代码:
package cn.itsource.sington;
/**
* 懒汉式
* @author Administrator
*/
public class SingletonTest2 {
/**
* 需求:单例模式 设计一个类 ,让这个 类中只存在一个对象的实例
*
* 1 需要一个私有的 构造方法,这样可以避免外部创建对象
*/
private SingletonTest2() {
}
/*
* 2 类加载进内存的时候,对象还没有存在,只有调用了getInstance()方法时,对
象才开始创建
*/
private static SingletonTest2 instance;
/*
* 3 向外公布一个 公共的 静态的方法 ,返回内部保存的这个对象
* 懒汉式是延迟加载,如果多个线程同时操作懒汉式时就有可能出现线程安全问题,
解决线程安全问题:
可以加同步来解决。但是加了同步之后,每一次都要比较锁,效率就变慢了,
所以可以加双重判断来提高程序效率。
*/
public static SingletonTest2 getInstance() {
if (instance == null) {
synchronized (SingletonTest2.class) {
if (instance == null) {
instance = new SingletonTest2();
}
}
}
return instance;
}
}
双重判断,对于多线程开发者来说不会陌生,如代码中所示,我们进行了两次if (instance == null)检查,这样就可以保证线程安全了。这样,实例化代码只用执行一次,后面再次访问时,判断if (instance == null),直接return实例化对象。
优点:线程安全;延迟加载;效率较高。
饿汉式和懒汉式的区别:
1饿汉式是类一加载进内存就创建好了对象;
2懒汉式则是类加载进内存的时候,对象还没有存在,只有调用了getInstance()方法时,对象才开始创建。
3懒汉式是延迟加载,如果多个线程同时操作懒汉式时就有可能出现线程安全问题,解决线程安全问题可以加同步来解决。但是加了同步之后,每一次都要比较锁,效率就变慢了,所以可以加双重判断来提高程序效率。
第四种实现方式:枚举的方式
示例代码:
package cn.itsource.sington;
public enum SingletonTest4 {
INSTANCE;
}
借助JDK1.5中添加的枚举来实现单例模式。不仅能避免多线程同步问题,而且还能防止反序列化重新创建新的对象。可能是因为枚举在JDK1.5中才添加,所以在实际项目开发中,很少见人这么写过。
优点:写法简单这是它最大的优点,其次可以自己处理序列化,是线程安全的
缺点:当想实例化一个单例类的时候,必须要记住使用SingletonTest4.INSTANCE获取对象的方法,而不是使用new,可能会给其他开发人员造成困扰,特别是看不到源码的时候。
以上是对单例模式常见的几种实现方式,在教学和学习过程中的一点简单的总结,希望对大家学习单例模式有一点点帮助。
Java单例模式几种实现方式的更多相关文章
- Java 8 的新特性和Java 的4种引用方式
一.接口的增强 Java 8允许我们给接口添加一个非抽象的方法实现,只需要使用 default关键字即可,这个特征又叫做扩展方法,示例如下: interface Formula { double ca ...
- java的两种同步方式, Synchronized与ReentrantLock的区别
java在编写多线程程序时,为了保证线程安全,需要对数据同步,经常用到两种同步方式就是Synchronized和重入锁ReentrantLock. 相似点: 这两种同步方式有很多相似之处,它们都是加锁 ...
- java集合四种遍历方式
package conection; import java.util.Iterator;import java.util.LinkedList;import java.util.List; publ ...
- Java多线程--两种实现方式
进程概述: 在这之前,有必要了解一下什么是进程? 在一个操作系统中,每个独立的执行的程序都可称为一个进程,也就是"正在运行的程序".如图所示: 线程概述: 如上所述,每个运行的程序 ...
- 转载:Java的四种引用方式
原文:https://www.cnblogs.com/huajiezh/p/5835618.html Java内存管理分为内存分配和内存回收,都不需要程序员负责,垃圾回收的机制主要是看对象是否有引用指 ...
- Java HashMap两种遍历方式
第一种: Map map = new HashMap(); Iterator iter = map.entrySet().iterator(); while (iter.hasNext()) { Ma ...
- JAVA多线程三种实现方式
JAVA多线程实现方式主要有三种:继承Thread类.实现Runnable接口.使用ExecutorService.Callable.Future实现有返回结果的多线程.其中前两种方式线程执行完后都没 ...
- Java 多线程 三种实现方式
Java多线程实现方式主要有三种:继承Thread类.实现Runnable接口.使用ExecutorService.Callable.Future实现有返回结果的多线程.其中前两种方式线程执行完后都没 ...
- 在java 中一种简单方式的声明静态Map常量的方法
我现在需要在一个类里面放一个HashMap,往里面放一些数据,每次要从数据库中取数据的时候先查找HashMap,看是否已经存在,若存在就直接提取,若不存在就从数据库中抽取数据之后再放到HashMap中 ...
随机推荐
- vscode 实用的插件
REST-Client api接口测试插件 在项目中新建以.http后缀名的文件即可. 右键使用 可以生产多种语言的请求代码块.可以说是非常舒服了. 使用代码块功能生成了node环境的http请求. ...
- 01Jenkins环境准备
01.硬件环境 Linux 64位(windows环境类似) 02.安装JDK a) 标题链接到Oracle官网下载jdk-8u144-linux-x64.rpm b) 将jdk拷贝到Linux的/t ...
- Spring Data MongoDB 分页查询
在上篇文章 Spring Data MongoDB 环境搭建 基础上进行分页查询 定义公用分页参数类,实现 Pageable 接口 import java.io.Serializable; impor ...
- 解决Windows资源管理器右键菜单打开EditPlus容易导致资源管理器无响应问题
这个问题确实很烦人,经常导致 资源管理器无响应,关闭后整个资源管理器彻底崩溃,原因未知.本着折腾和强迫症精神,这个问题一定要解决.方法如下: 1.不要用EditPlus自带的添加到系统右键菜单选项. ...
- 微信小程序开发11-HTTPS网络通信(重点)
1.OneNET平台支持https,将HTTP头部改成https://api.heclouds.com即可(重点!!!!!!!!) 2.如果我们需要从 https://test.com/getinfo ...
- IEC62304软件维护框架
软件维护计划的任务 建立接收.记录.评估.解决和追踪医疗器械软件发行后的反馈 制定确认反馈是否是问题的标准 使用风险管理过程 使用配置管理过程 制定升级.补丁以及遗留问题修正计划 问题和修改分析的任务 ...
- 常用sql commands以及mysql问题解决日志
mysql workbench常用命令快捷键 ctrl+T ->创建新的sql query tab ctrl+shift+enter->执行当前的sql命令 https://dev.mys ...
- [NCH 1, 3]
Preview: 1. Implement strStr() O(m*n): class Solution { public: int strStr(string haystack,string ne ...
- Python super执行基类的构造方法
有一个需求是这样的,先有一个Machine的类,有通用的name.position.date属性,然后又出现了一个Server的类继承Machine类,但是Server类多出一个ipaddr的属性,所 ...
- MP4个人制作