单例模式
核心作用
- 保证一个类中只有一个实例,并且提供一个访问该实例的全局访问点
- 项目中的配置文件,一般也只有一个对象,没必要每次都都new一遍,配置文件不容易改变
- Spring中bean的加载,以及控制器对象也是单例模式
单例模式五种实现方式
- 饿汗式:编程安全,调用效率高。但是不能延时加载
- 懒汉式:线程安全,调用效率不高,但是可以延时加载
- 双重检测锁式(由于JVM底层内部模型原因,偶尔出现问题,不建议使用)
- 静态内部类式:线程安全,调用效率高,但是可以延时加载
- 枚举单例:线程安全,调用效率高,不能延时加载
代码
饿汗式 :线程安全的,类初始化时立即加载,天然线程安全
public class SingleEHan {
private static SingleEHan instance = new SingleEHan(); //类实例化,立即加载 //,天然线程安全
private SingleEHan(){} //单例模式构造器方法必须私有
public static SingleEHan getInstance(){ //不需要同步,显然调用的效率高
return instance;
}
}
Compiled from "SingleEHan.java"
public class com.xhd.admin.web.SingleEHan {
public static com.xhd.admin.web.SingleEHan getInstance();
Code:
0: getstatic #2 // Field instance:Lcom/xhd/admin/web/SingleEHan;
3: areturn
static {};
Code:
0: new #3 // class com/xhd/admin/web/SingleEHan
3: dup
4: invokespecial #4 // Method "<init>":()V
7: putstatic #2 // Field instance:Lcom/xhd/admin/web/SingleEHan;
10: return
}
懒汉式加载 资源利用效率高,但调用效率低
public class SingleLanHan {
private static SingleLanHan singleLanHan;
private SingleLanHan(){
if(singleLanHan != null){ //防止反射
throw new RuntimeException();
}
}
// synchronized 调用效率低
public static synchronized SingleLanHan getSingleLanHan(){
if(singleLanHan == null){
singleLanHan = new SingleLanHan();
}
return singleLanHan;
}
}
静态内部类实现方式:,线程安全调用效率高 懒加载
public class SingleStatic {
private SingleStatic() {}
//静态内部类
private static class SingleStaticInstance{
private static final SingleStatic instance
= new SingleStatic();
}
public static SingleStatic getInstance(){
return SingleStaticInstance.instance;
}
}
Compiled from "SingleStatic.java"
public class com.xhd.admin.web.SingleStatic {
public static com.xhd.admin.web.SingleStatic getInstance();
Code:
0: invokestatic #3
// Method com/xhd/admin/web/SingleStatic$SingleStaticInstance.access$100:()Lcom/xhd/admin/web/SingleStatic;
3: areturn
com.xhd.admin.web.SingleStatic(com.xhd.admin.web.SingleStatic$1);
Code:
0: aload_0
1: invokespecial #1 // Method "<init>":()V
4: return
}
枚举:天然线程安全,天然实例唯一
public enum SingleEnum {
//这个枚举元素,本身就是单例对象
INSTANCE;
//添加自己需要的操作
public void singletonOperation(){
}
}
双重检查锁实现单例模式
public class SingleDoubleCheck {
private static SingleDoubleCheck singleDoubleCheck = null;
private SingleDoubleCheck(){}
public static SingleDoubleCheck getSingleDoubleCheck(){
if(singleDoubleCheck == null){
SingleDoubleCheck sc ;
synchronized (SingleDoubleCheck.class){
sc = singleDoubleCheck;
if(sc == null){
sc = new SingleDoubleCheck();
}
}
singleDoubleCheck = sc;
}
return singleDoubleCheck;
}
}
多线程环境下测试用CountDownLatch
- 同步辅助类,在完成一组正在其他线程中执行的操作之前,它允许一个或多个线程一直等待
- countDown() 当前线程调此方法,则统计减一,建议放在finally里执行
- await方法:调用此方法会一直阻塞当前线程,直到计数器的值为0
public class ClientTest {
public static void main(String[] args) throws InterruptedException {
long start = System.currentTimeMillis();
int threadNum = 10;
final CountDownLatch countDownLatch = new CountDownLatch(threadNum);
for(int j = 0; j< threadNum;j++){
new Thread(new Runnable() {
@Override
public void run() {
for(int i = 0;i<1000000;i++){
Object o = SingleEHan.getInstance(); //换方式在此地方
countDownLatch.countDown();
}
}
}).start();
}
countDownLatch.await(); //await 自动检测
long end = System.currentTimeMillis();
long between = end - start;
System.out.println("总耗时:"+between);
}
}
序列化与反序列化问题,以及反射问题
public class SingleLanHanSerial implements Serializable{
private static SingleLanHanSerial singleLanHan;
private SingleLanHanSerial(){
if(singleLanHan != null){ //防止反射
throw new RuntimeException();
}
}
public static synchronized SingleLanHanSerial getSingleLanHan(){
// synchronized 调用效率低
if(singleLanHan == null){
singleLanHan = new SingleLanHanSerial();
}
return singleLanHan;
}
// 反序列化,如果定义了readResolve方法,就直接返回Instance实例,防止了反序列化的漏洞
private Object readResolve() throws ObjectStreamException {
return singleLanHan;
}
}
反射测试序列化测试
public class ClientEnum {
public static void main(String[] args) throws Exception {
//通过反射机制创造对象
Class<SingleLanHan> lanHanClazz = (Class<SingleLanHan>)Class.forName("com.jenny.mole.single.SingleLanHan");
Constructor<SingleLanHan> constructor = lanHanClazz.getDeclaredConstructor(null);
constructor.setAccessible(true); //可以访问私有方法了这样
SingleLanHan singleLanHan = constructor.newInstance();
SingleLanHan singleLanHan1 = constructor.newInstance();
System.out.println(singleLanHan);
System.out.println(singleLanHan1);
// 通过反序列化的方式创造对象
SingleLanHanSerial single2 = SingleLanHanSerial.getSingleLanHan();
FileOutputStream fos = new FileOutputStream("/d:/a.txt");
ObjectOutputStream oos = new ObjectOutputStream(fos);
oos.writeObject(single2);
ObjectInputStream ois = new
ObjectInputStream(new FileInputStream("/d:/a.txt"));
SingleLanHanSerial s3 = (SingleLanHanSerial)ois.readObject();
System.out.println(single2);
System.out.println(s3);
}
}
静态内部类
外部类没有static属性,则不会像懒汉式那样立即加载对象只有真正调用getInstance,才会加载静态内部类,加载类时时线程安全的,Instance是static final类型,保证了内存中只有一个实例存在,而且只能被赋值一次,从而保证了线程安全性兼备了并发高效调用和延迟加载的优势
枚举方式
-
优点:实现简单,枚举本身就是单例模式,由JVM从根本上提供保障,避免通过反射和反序列化的漏洞
- 缺点:无延迟加载
如何选用
-
单例对象,占用资源少,不需要延时加载:枚举好于 懒汉式
-
单例对象占用资源大,需要延时加载:静态内部类好于懒汉式
分享到:
相关推荐
Java设计模式之单例模式的七种写法 单例模式是一种常见的设计模式,它确保某个类只有一个实例,而且自行实例化并向整个系统提供这个实例。在计算机系统中,线程池、缓存、日志对象、对话框、打印机的驱动程序对象常...
python 设计模式之单例模式
设计模式之单例模式详解 单例模式是一种常用的软件设计模式,其定义是单例对象的类只能允许一个实例存在。许多时候整个系统只需要拥有一个的全局对象,这样有利于我们协调系统整体的行为。 单例模式的实现主要是...
通过研磨设计模式之单例模式的资料,你可以深入理解单例模式的原理、实现方式及其优缺点,进一步提升自己的编程技能和设计思维。学习并熟练掌握设计模式,对于成为一名优秀的Java开发者至关重要。
Java设计模式之单例模式详解 一、单例模式概览 单例模式(Singleton Pattern)是面向对象设计模式中的一种,属于创建型模式。它确保一个类仅有一个实例,并提供一个全局访问点来访问该实例。单例模式的核心在于控制...
单例模式是软件设计模式中的一种经典模式,它保证了类只有一个实例存在,并提供一个全局访问点。在Java等面向对象编程语言中,单例模式常用于管理共享资源,如数据库连接池、线程池或者配置文件等。结合工厂模式,...
### PHP设计模式之单例模式详解 #### 一、引言 在软件工程领域,设计模式是一种被广泛接受的解决方案,用于解决特定类型的问题。PHP作为一种流行的服务器端脚本语言,同样可以从这些设计模式中受益。本文将详细介绍...
JAVA设计模式之单例模式(完整版)1[定义].pdf
单例模式是一种常用的设计模式,它的核心思想是在整个应用程序中,一个类只能有一个实例存在。单例模式常用于控制资源的共享,例如数据库连接池、日志服务等。单例模式有多种实现方式,常见的包括懒汉式、饿汉式以及...
细心整合和单例模式和工厂模式的几种模型,懒汉式,饿汉式,如何并发操作模式,等都有详细讲解
设计模式之单例模式,单列模式的几种实现形式,以及其优缺点,还有就是示例,对初步了解单列模式的有所帮助
单例模式是一种常用的软件设计模式,其定义是单例对象的类只能允许一个实例存在。 许多时候整个系统只需要拥有一个的全局对象,这样有利于我们协调系统整体的行为。 比如在某个服务器程序中,该服务器的配置信息存放...
本次我们将深入探讨两种设计模式——单例模式和装饰模式,它们在Java编程中都有着广泛的应用。 首先,让我们来理解“单例模式”。单例模式是一种创建型设计模式,其核心思想是保证一个类只有一个实例,并提供一个...
目录 单例模式的概念 单例模式的要点 单例模式类图 单例模式归类 单例模式的应用场景 单例模式解决的问题 单例模式的实现方式 单例模式实现方式对比 单例模式的概念 单例模式,顾名思义就是只有一个实例,并且由它...