设计模式(Java语言)-单例模式
单例模式,简而言之就是在整个应用程序里面有且仅有一个实例,在程序的任何时候,任何地方获取到的该对象都是同一个对象。单例模式解决了一个全局的类被频繁创建和销毁的,或者每次创建或销毁都需要消耗大量cpu资源的对象的问题。单例模式总的可以分为懒汉模式和饿汉模式,顾名思义,懒汉模式是一个非常懒的汉子,只要你没有使用到它,它就永远不会实例化。饿汉模式的意思就是,汉子非常饥渴,只要在程序的编译阶段就给你分配内存,创建好对象。
将懒汉模式和饿汉模式细分,又可以分为:
1、懒汉模式
2、饿汉模式
3、双检模式
4、静态内部类模式
5、枚举模式
不管是用哪一种方式实现的单例模式,其创建流程基本都是一直的:首先将构造方法声明为private的,这样就防止直接new出一个新的对象。第二,声明一个私有的成员变量,即单例对象。第三步,声明一个public的静态方法,用于获取或创建单例对象,外部想要获取该对象必须通过这个方法获取。
一、懒汉模式1--线程安全
/**
* 饿汉模式1
*/
public class HungrySingleton1 { private static HungrySingleton1 singleton = new HungrySingleton1(); private HungrySingleton1(){} public static HungrySingleton1 getInstance() {
return singleton;
} }
这种懒汉模式的优点是实现非常简单。缺点是并起到懒加载的效果,如果项目没有使用到这个对象的就会造成资源的浪费。
二、饿汉模式1--线程不安全
/**
* 懒汉模式1
*/
public class LazySingleton1 { private static LazySingleton1 singleton; private LazySingleton1(){} public static LazySingleton1 getInstance() {
if (singleton == null) {
singleton = new LazySingleton1();
}
return singleton;
} }
这种写法虽然实现了懒加载的效果,但是严格意义上并不是单例模式,因为在多线程的环境下有可能会创建出多个不同的对象,至于为什么,不懂的可以看一下我之间写的关于Java内存模型的文章。这种写法只能应用于单线程的环境下,局限性很大。实际中强烈不建议使用这种方法。
三、懒汉模式2--线程安全
/**
* 懒汉模式2
*/
public class LazySingleton2 { private static LazySingleton2 singleton; private LazySingleton2(){} public static synchronized LazySingleton2 getInstance() {
if (singleton == null) {
singleton = new LazySingleton2();
}
return singleton;
} }
这种写法咋看跟上面的方法一样,这种写法在方法上添加了 synchronized 关键字,这样就保证了每次只能有一个线程进入方法体中,解决了懒汉模式1中出现的问题。这种写法的优点是实现了懒加载的效果,缺点是效率非常低,当多个线程同时获取实例时,有可能会造成线程阻塞的情况。不推荐使用。
懒汉模式3--线程不安全
/**
* 懒汉模式3
*/
public class LazySingleton3 { private static LazySingleton3 singleton; private LazySingleton3(){} public static LazySingleton3 getInstance() {
if (singleton == null) {
synchronized (LazySingleton3.class) {
if (singleton == null) {
singleton = new LazySingleton3();
}
}
}
return singleton;
}
}
这种写法进行了两次 singleton == null 的判断,在实际的应用中当我们调用这个方法时,其实99%的几率是实例就已经创建好了,因此第一个 singleton == null 能过滤掉99%的调用,不用将方法锁起来,从而提高了效率。这种方法的优点是实现懒加载的效果,效率和很高。缺点是代码设计仍然后缺陷,jvm在为对象分配内存和赋值并不是一个原子操作,即 singleton = new LazySingleton3() 这段代码在jvm中是由三个步骤实现的,首先jvm会在堆中为对象分配一定的内存空间,然后完成对象的初始化工作,然后将内存地址指向到对象中。但是,我们知道,jvm在编译的时候并不总是根据我们编写的代码的顺序来执行了,而是根据jvm觉得最优的顺序执行(这个过程就叫做指令重排序),所以有可能在执行了步骤1后就执行了步骤3,这时候第二个线程进来的发现singleton并不为空,因此就直接返回了该对象,因此造成空指针异常。
四、双重检查锁模式---线程安全
/**
* 懒汉模式4
*/
public class LazySingleton4 { private volatile static LazySingleton4 singleton; private LazySingleton4(){} public static LazySingleton4 getInstance() {
if (singleton == null) {
synchronized (LazySingleton4.class) {
if (singleton == null) {
singleton = new LazySingleton4();
}
}
}
return singleton;
}
}
相较于上面的方式,这种方式只是在成员变量中添加了 volatile 关键字,解决了指令重排序的问题,同时确保当前线程修改了这个变量时,其他的线程能够及时读到最新的值。这种方法缺点是写起来比较复杂,要求程序员对jvm比较理解。优点是既保证了线程安全,同时也能够保证了比较高的效率。
五、静态内部类模式--线程安全
/**
* 懒汉模式5
*/
public class LazySingleton5 { private LazySingleton5(){} private static class Holder {
private static final LazySingleton5 INSTANCE = new LazySingleton5();
} public static LazySingleton5 getInstance() {
return Holder.INSTANCE;
} }
这种写法实现比较简单,即实现了懒加载的效果,同时也保证的多线程环境下的线程安全问题。推荐使用这种方式。
六、枚举模式 -- 线程安全
/**
* 懒汉模式6
*/
public enum LazySingleton6 { INSTANCE } //使用方法
public class Test { public static void main(String[] args) {
LazySingleton6 instance = LazySingleton6.INSTANCE;
LazySingleton6 instance1 = LazySingleton6.INSTANCE;
System.out.println(instance == instance1); } }
推荐写法,简单高效。充分利用枚举类的特性,只定义了一个实例,且枚举类是天然支持多线程的。
喜欢我写的博客的同学可以关注订阅号【Java解忧杂货铺】,里面不定期发布一些技术干活,也可以免费获取大量最新最流行的技术教学视频
设计模式(Java语言)-单例模式的更多相关文章
- 设计模式 - Java中单例模式的6种写法及优缺点对比
目录 1 为什么要用单例模式 1.1 什么是单例模式 1.2 单例模式的思路和优势 2 写法① - 饥饿模式 2.1 代码示例 2.2 优缺点比较 3 写法② - 懒惰模式 3.1 代码示例 3.2 ...
- 10.Java设计模式 工厂模式,单例模式
Java 之工厂方法和抽象工厂模式 1. 概念 工厂方法:一抽象产品类派生出多个具体产品类:一抽象工厂类派生出多个具体工厂类:每个具体工厂类只能创建一个具体产品类的实例. 即定义一个创建对象的接口(即 ...
- Java设计模式之《单例模式》及应用场景
摘要: 原创作品,可以转载,但是请标注出处地址:http://www.cnblogs.com/V1haoge/p/6510196.html 所谓单例,指的就是单实例,有且仅有一个类实例,这个单例不应该 ...
- Java设计模式之【单例模式】
Java设计模式之[单例模式] 何为单例 在应用的生存周期中,一个类的实例有且仅有一个 当在一些业务中需要规定某个类的实例有且仅有一个时,就可以用单例模式 比如spring容器默认初始化的实例就是单例 ...
- Java设计模式中的单例模式
有时候在实际项目的开发中,我们会碰到这样一种情况,该类只允许存在一个实例化的对象,不允许存在一个以上的实例化对象,我们将这种情况称为Java设计模式中的单例模式.设计单例模式主要采用了Java的pri ...
- 设计模式(Java语言)- 工厂方法模式
前言 在介绍工厂方法模式之前,我们需要知道这个设计模式是什么,解决了什么样的问题?在上一篇博客 设计模式(Java语言)- 简单工厂模式 介绍了简单工厂模式,然后总结了简单工厂模式的缺点: 1.当新增 ...
- 设计模式(Java语言)- 原型模式
原型模式(Prototype Pattern)也有人将原型模式称为克隆模式,是属于创造型设计模式,用于创建重复的对象,提供了一种创建对象的最佳方式.原型模式需要实现Cloneable接口,来实现对象的 ...
- java 之 单例模式(大话设计模式)
笔者记得去面试时曾被问起这个模式,当时还没有看过设计模式,对设计模式基本上一无所知,不过可以肯定的是笔者用过单例模式.当时回答的风马牛不相及,很尴尬. 也是从那时起,开始学习设计模式.今天所说的就是单 ...
- 2019 Android 高级面试题总结 从java语言到AIDL使用与原理
说下你所知道的设计模式与使用场景 a.建造者模式: 将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示. 使用场景比如最常见的AlertDialog,拿我们开发过程中举例,比如C ...
- 六、java基础-单例模式_继承_覆盖_多态
1.单例模式: 1)提出原因 是由gof 也就是四人组提出来的.为了保证jvm中某一类型的java对象永远只有一个,同时也是为了节省内存的开销.因为外面程序可以通过new的方法直接调用类里面的构造方法 ...
随机推荐
- Hadoop实战1:MapR在ubuntu集群中的安装
由于机器学习算法在处理大数据处理的时候在所难免的会效率降低,公司需要搭建hadoop集群,最后采用了商业版的Hadoop2(MapR). 官网: http://doc.mapr.com/display ...
- HTML innerHTML、textContext、innerText
网址 : https://developer.mozilla.org/en-US/docs/Web/API/Element/innerHTML 1.innerHTML : 获得.修改元素的用HTML语 ...
- MySQL 数据库的设计规范
网址 :http://blog.csdn.net/yjjm1990/article/details/7525811 1.文档的建立日期.所属的单位.2.数据库的命名规范.视图.3.命名的规范:1)避免 ...
- 17.Linux搭建网络仓库
1.搭建一个网络仓库 服务端:10.0.0.201 1.准备软件包(1.光盘 2.缓存 3.联网下载 4.同步) 1.挂载光盘 mount /dev/cdrom 2.通过ftp共享软件包存放的目录 y ...
- 每日温度(LeetCode Medium难度算法题)题解
LeetCode 题号739中等难度 每日温度 题目描述: 根据每日 气温 列表,请重新生成一个列表,对应位置的输入是你需要再等待多久温度才会升高超过该日的天数.如果之后都不会升高,请在该位置用 0 ...
- Yii2.0教程应用结构篇 —— 入口脚本
入口脚本是应用启动流程中的第一环,一个应用(不管是网页应用还是控制台应用)只有一个入口脚本.终端用户的请求通过入口脚本实例化应用并将将请求转发到应用. Web 应用的入口脚本必须放在终端用户能够访问的 ...
- ArcGIS Engine简单图形绘制功能的实现(点、线、面)
我们添加点.线.面来实现图形的编辑需要使用Geometry对象类. Point(点) 是一个0维的几何图形,具有X.Y坐标值,以及可选的属性,如高程值(Z值).度量值(M值).ID值等,可用于描述需要 ...
- 你的 Java 并发程序 Bug,100% 是这几个原因造成的
可见性问题 可见性是指一个线程对共享变量进行了修改,其他线程能够立马看到该共享变量更新后的值,这视乎是一个合情合理的要求,但是在多线程的情况下,可能就要让你失望了,由于每个 CPU 都有自己的缓存,每 ...
- JAVA nio 简单使用
nio 模拟客户端和服务器互相通讯--传输一个int值,并且不断的+1: 服务器,单线程 public class Server { public static void main(String[] ...
- [NOI2001]食物链(并查集拓展域)&& [HAOI2006]旅行(Kruskal)
题目描述 动物王国中有三类动物 A,B,C,这三类动物的食物链构成了有趣的环形.A 吃 B,B 吃 C,C 吃 A. 现有 N 个动物,以 1 - N 编号.每个动物都是 A,B,C 中的一种,但是我 ...