java设计模式——单例模式(一)
一. 定义与类型
定义:保证一个类仅有一个实例,并提供一个全局访问点
类型:创建型
二. 适用场景
想确保任何情况下都绝对只用一个实例
三. 优缺点
优点:
在内存里只有一个实例,减少了内存开销
可以避免对资源的多重占用
设置全局访问点,严格控制访问
缺点:
没有接口,扩展困难
四. 重点
私有构造器
线程安全
延迟加载
序列化和反序列化安全
反射
实用技能:反编译,内存原理,多线程Debug
五. 相关设计模式
单例模式和工厂模式
单例模式和享元模式
六. Coding
懒汉式:
/**
* @program: designModel
* @description: 懒汉单例,懒汉式注重的就是延迟加载,当在使用到这个实例的时候才会初始化
* @author: YuKai Fan
* @create: 2018-12-04 14:04
**/
public class LazySingleton {
private static LazySingleton lazySingleton = null;
private LazySingleton() { }
//如果只有这种方法,是线程不安全的。
//在多线程环境下,使用这个方法,会有概率的产生不止一个实例的情况,虽然最后返回的还是同样的独享
/*public static LazySingleton getInstance() {
if (lazySingleton == null) {
lazySingleton = new LazySingleton();
}
return lazySingleton;
}*/ //在方法上加锁,让这个方法每次只能有一个线程访问
/*public synchronized static LazySingleton getInstance() {
if (lazySingleton == null) {
lazySingleton = new LazySingleton();
}
return lazySingleton;
}*/ //在代码块上加锁,让这个方法每次只能有一个线程访问,这样只会产生一个实例
//这种方式,锁的是class类,存在加锁和解锁的开销,对性能有一定影响
public static LazySingleton getInstance() {
synchronized(LazySingleton.class) {
if (lazySingleton == null) {
lazySingleton = new LazySingleton();
}
}
return lazySingleton;
}
}
如上方代码注释所说,懒汉式的注重点是延时加载,但是在性能和安全方面都所有影响,所以引出下面的双重检查模式
/**
* @program: designModel
* @description: 懒汉式双重检查。既符合延迟加载,也保证了安全性能
* @author: YuKai Fan
* @create: 2018-12-04 14:35
**/
public class LazyDoubleCheckSingleton {
//volatile关键字,使用解决了程序的重排序问题,即使在多线程的情况下,实例变量始终是最新的状态
//使用这种方法是不让下面的2,3发生重排序
private volatile static LazyDoubleCheckSingleton lazyDoubleCheckSingleton = null;
private LazyDoubleCheckSingleton() { }
public static LazyDoubleCheckSingleton getInstance() {
if (lazyDoubleCheckSingleton == null) {
synchronized (LazyDoubleCheckSingleton.class) {
if (lazyDoubleCheckSingleton == null) {
//1.分配内存给这个对象
//2.初始化对象
//3.设置LazyDoubleCheckSingleton 只向刚分配的内存地址
//其中2和3的顺序可能会被颠倒,倒置,判断为null的时候,已经初始化对象,此时并不为空
//intra-thread semantics 不会改变单线程程序的重排序
lazyDoubleCheckSingleton = new LazyDoubleCheckSingleton();
}
}
}
return lazyDoubleCheckSingleton;
}
}
双重检查,顾名思义是对实例的是否创建做了两层的判断,例如,当thread0,进入getInstance(),此时lazyDoubleCheckSingleton==null,两层的判断都会过,当刚刚走到new LazyDoubleCheckSingleton()时,thread1进入方法时,第一次判断其实lazyDoubleCheckSingleton还是为null,所以会通过,但是由于后面代码加锁,所以无法进入,当thread0执行玩方法时,释放锁,让thread1拿到,在进行判断,此时lazyDoubleCheckSingleton != null,就会直接退出,这样保证了在多线程环境下,始终只有一个实例。
刚开始的时候,我一直认为这样多此一举,为什么要在加锁的外面在判断一次。后来思考明白了,thread0,thread1的执行顺序,以及执行方法的时机在真正的环境下其实是不知道的,由操作系统来决定的。所以有可能是在thread0已经拿到实例的时候,此时thread1才进入方法,这时候这个判断就会不成立,也就不会在获取锁和释放锁,也就提高了程序的性能。
从上面的代码注释可知,这种方式存在重排序的问题,上面解决重排序的问题是使用volatile关键字,来防止重排序,还有一种方式是可以重排序,但是thread1,不会看到。
/**
* @program: designModel
* @description: 使用静态内部类来防止多线程环境下对DoubleCheck的懒汉式单例模式的判断实例问题,同时也允许重排序
* @author: YuKai Fan
* @create: 2018-12-04 16:35
**/
public class StaticInnerClassSingleton {
//使用静态内部类,是基于class对象的初始化锁的延迟加载方式,在线程0执行方法时会去实例,即使发生重排序,线程1也会被锁在初始化阶段
private static class InnerClass{
private static StaticInnerClassSingleton staticInnerClassSingleton = new StaticInnerClassSingleton();
}
public static StaticInnerClassSingleton getInstance() {
return InnerClass.staticInnerClassSingleton;
}
private StaticInnerClassSingleton() { }
}
饿汉式:
这是一个很简单的单例模式的方式,它与懒汉式的最大的区别就是延时加载。饿汉式单例模式是在程序加载的时候,就创建实例,但是如果不用该实例,就会占用资源
/**
* @program: designModel
* @description: 饿汉式,与懒汉式最大的区别,就是延时加载,但是饿汉式如果不用该实例,会占用资源
* @author: YuKai Fan
* @create: 2018-12-04 16:57
**/
public class HungrySingleton implements Serializable {
private final static HungrySingleton hungrySingleton; static {
hungrySingleton = new HungrySingleton();
}
private HungrySingleton() { }
public static HungrySingleton getInstance() {
return hungrySingleton;
} private Object readResolve() {
return hungrySingleton;
}
}
单线程的执行顺序
多线程的执行顺序
java设计模式——单例模式(一)的更多相关文章
- java设计模式单例模式 ----懒汉式与饿汉式的区别
常用的五种单例模式实现方式 ——主要: 1.饿汉式(线程安全,调用率高,但是,不能延迟加载.) 2.懒汉式(线程安全,调用效率不高,可以延时加载.) ——其他: 1.双重检测锁式(由于JVM底层内部模 ...
- Java设计模式の单例模式
-------------------------------------------------- 目录 1.定义 2.常见的集中单例实现 a.饿汉式,线程安全 但效率比较低 b.单例模式的实现:饱 ...
- JAVA设计模式-单例模式(Singleton)线程安全与效率
一,前言 单例模式详细大家都已经非常熟悉了,在文章单例模式的八种写法比较中,对单例模式的概念以及使用场景都做了很不错的说明.请在阅读本文之前,阅读一下这篇文章,因为本文就是按照这篇文章中的八种单例模式 ...
- Java设计模式 - - 单例模式 装饰者模式
Java设计模式 单例模式 装饰者模式 作者 : Stanley 罗昊 [转载请注明出处和署名,谢谢!] 静态代理模式:https://www.cnblogs.com/StanleyBlogs/p/1 ...
- 【设计模式】Java设计模式 - 单例模式
[设计模式]Java设计模式 - 单例模式 不断学习才是王道 继续踏上学习之路,学之分享笔记 总有一天我也能像各位大佬一样 分享学习心得,欢迎指正,大家一起学习成长! 原创作品,更多关注我CSDN: ...
- Java 设计模式 —— 单例模式
1. 概念: 单例模式是一种常用的软件设计模式.核心结构中只包含一个被称为单例的特殊类.通过单例模式可以保证系统中一个类只有一个实例而且该实例易于外界访问,从而方便对实例个数的控制并节约系统资源.如果 ...
- Java设计模式 - 单例模式 (懒汉方式和饿汉方式)
概念: Java中单例模式是一种常见的设计模式,单例模式的意思就是只有一个实例.单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例.这个类称为单例类. 单例模式的写法有好几种,这 ...
- JAVA设计模式--单例模式
单例设计模式 Singleton是一种创建型模式,指某个类采用Singleton模式,则在这个类被创建后,只可能产生一个实例供外部访问,并且提供一个全局的访问点. 核心知识点如下: (1) 将采用单例 ...
- Java设计模式-单例模式(Singleton)
单例对象(Singleton)是一种常用的设计模式.在Java应用中,单例对象能保证在一个JVM中,该对象只有一个实例存在.这样的模式有几个好处: 1.某些类创建比较频繁,对于一些大型的对象,这是一笔 ...
随机推荐
- C#网络编程学习(4)---Socket Tcp进阶之 使用异步循环接收客户端连接和信息
1.方法介绍 BeginAccept(AsyncCallback callback, object state); 异步开始监听客户端连接. callback为一个委托,在成功接收客户端连接时调用委托 ...
- 洛谷P1318 积水面积
题目描述 一组正整数,分别表示由正方体叠起的柱子的高度.若某高度值为\(x\),表示由\(x\)个正立方的方块迭起(如下图,\(0<=x<=5000\)).找出所有可能积水的地方(图中蓝色 ...
- 项目 06 Bootstrap
项目班 06 Bootstrap 一.介绍 #基于HTML,CSS,JS的简洁灵活的流行前端框架及交互组件集 #为快速WEB开发提供了一套前端工具包,包括布局.网格.表格.按钮.表单.导航.提示等等 ...
- 简述raid0,raid1,raid5,raid10 的工作原理及特点
RAID 0 支持1块盘到多块盘,容量是所有盘之和 RAID1 只支持2块盘,容量损失一块盘 RAID 5最少三块盘,不管硬盘数量多少,只损失一块容量 RAID 10最少4块盘,必须偶数硬盘,不管硬盘 ...
- 安装mongodb并配置
下载网址http://dl.mongodb.org/dl/win32/x86_64 mongodb-win32-x86_64-2008plus-ssl-v3.4-latest.zip 解压d盘命名mo ...
- GUI的最终选择 Tkinter(三):Checkbutton组件和Radiobutton组件、LabelFrame组件
Checkbutton组件 Checkbutton组件就是常见的多选按钮,而Radiobutton则是单选按钮 from tkinter import * root = Tk() v = IntVar ...
- Selenium----Selenium简单介绍以及Selenium IDE环境搭建,脚本录制
1.selenium简单介绍 心得:作为一个新手开始了解这个工具,打算从录制脚本开始学习,“录制,看,学习,写”,总结网友说得打算先使用Selenium IDE录制学习,再使用Selenium RC开 ...
- Washing Plates 贪心
https://www.hackerrank.com/contests/101hack41/challenges/washing-plates 给定n个物品,选这个物品,贡献 + p, 不选的话,贡献 ...
- 基于nginx的FastCGI的缓存配置
废话不多说了, 直接上配置, 其实 fastcgi_cache 和 proxy_cache 的配置基本一样: # !缓存文件存放目录 # levels 缓存层次 # keys_zone 缓存空间名和共 ...
- 在sublime text3下,用快捷键把文件打开到浏览器中
使用背景 在编辑html或者js文件的时候,是否想在浏览器中预览一下, 你的步骤可能是这样的: 找到编辑文件的位置, 右键使用某一浏览器打开.如果是这样,你就out了, 因为在sublime中有更加简 ...