`
Franciswmf
  • 浏览: 799934 次
  • 性别: Icon_minigender_1
  • 来自: 上海
文章分类
社区版块
存档分类
最新评论

java单例模式的5种写法-饿汉、懒汉、静态内部类、双重校验锁、枚举

 
阅读更多
参考博客:
1.单例和多例的区别
http://www.cnblogs.com/zhangliang88/p/5388472.html
2.Java:单例模式的七种写法
http://www.blogjava.net/kenzhh/archive/2013/03/15/357824.html
3.Java Singleton
http://blog.csdn.net/penngrove/article/details/5753977
==============================================================================
【单例类的5种写法】
==============================================================================
1、饿汉方式
/**
 * 饿汉,初始化时即生成
 * @author franciswmf
 *这种方式基于classloder机制避免了多线程的同步问题,不过,instance在类装载时就实例化,
 *虽然导致类装载的原因有很多种,在单例模式中大多数都是调用getInstance方法, 
 *但是也不能确定有其他的方式(或者其他的静态方法)导致类装载,这时候初始化instance显然没有达到lazy loading的效果。
 */
public class TraditionalSingleton {
	private TraditionalSingleton(){
		System.out.println("TraditionalSingleton constructor execute");
	}
	private static TraditionalSingleton singleton=new TraditionalSingleton();
	
	public static TraditionalSingleton getInstance(){
		return singleton;
	}
}


2、懒汉方式
/**
 * 懒汉,线程安全(但效率低)
 * @author franciswmf
 *这种写法能够在多线程中很好的工作,而且看起来它也具备很好的lazy loading,
 *但是,遗憾的是,效率很低,99%情况下不需要同步。
 */
public class LazySingleton {
	private static LazySingleton instance;
	private LazySingleton(){
		System.out.println("LazySingleton constructor execute");
	}
	public static synchronized LazySingleton getInstance(){//加synchronized同步关键字
		if(null==instance){
			instance=new LazySingleton();
		}
		return instance;
	}
}


3、静态内部类方式
/**
 * 静态内部类方式
 * @author franciswmf
 *这种方式同样利用了classloder的机制来保证初始化instance时只有一个线程,
 *它跟第三种和第四种方式不同的是(很细微的差别):第三种和第四种方式是只要Singleton类被装载了,
 *那么instance就会被实例化(没有达到lazy loading效果),而这种方式是Singleton类被装载了,
 *instance不一定被初始化。因为SingletonHolder类没有被主动使用,只有显示通过调用getInstance方法时,
 *才会显示装载SingletonHolder类,从而实例化instance。想象一下,如果实例化instance很消耗资源,
 *我想让他延迟加载,另外一方面,我不希望在Singleton类加载时就实例化,因为我不能确保Singleton类还可能在其他的地方被主动使用从而被加载,
 *那么这个时候实例化instance显然是不合适的。这个时候,这种方式相比第三和第四种方式就显得很合理。

 */
public class StaticInnerClassSingleton {
	private StaticInnerClassSingleton(){
		System.out.println("StaticInnerClassSingleton constructor execute");
	}
	private static class InnerFactory{
		private static final StaticInnerClassSingleton instance=new StaticInnerClassSingleton();//内部类
	}
	
	public static StaticInnerClassSingleton getInstance(){
		return InnerFactory.instance;
	}
	
}


4、双重校验锁方式
/**
 * 双重校验锁
 * @author franciswmf
 *这个是Singleton3方式的升级版,俗称双重检查锁定,
 *详细介绍请查看:http://www.ibm.com/developerworks/cn/java/j-dcl.html
 *在JDK1.5之后,双重检查锁定才能够正常达到单例效果。
 */
public class DoubleCheckSingleton {
	
	private DoubleCheckSingleton(){
		System.out.println("DoubleCheckSingleton constructor execute");
	}
	private volatile static DoubleCheckSingleton singleton;
	//
	public static DoubleCheckSingleton getInstance(){
		if(singleton==null){
			synchronized(DoubleCheckSingleton.class){
				if(singleton==null){
					singleton=new DoubleCheckSingleton();
				}
			}
		}
		return singleton;
	}

}


5、枚举方式
/**
 * 枚举
 * 这种方式是Effective Java作者Josh Bloch 提倡的方式,它不仅能避免多线程同步问题,而且还能防止反序列化重新创建新的对象,
 * 可谓是很坚强的壁垒啊,不过,个人认为由于1.5中才加入enum特性,用这种方式写不免让人感觉生疏,在实际工作中,我也很少看见有人这么写过。
 * @author franciswmf
 *
 */
public enum EnumSingleton {
	INSTANCE;
	EnumSingleton(){
		System.out.println("EnumSingleton constructor execute");
	}

}


====================================================================
【用Reflection + SetAccessable()来攻击单例,如果能创建出两份单例,就认为攻击成功】
====================================================================
public class ReflectionAttack {

	/**
	 * 1-TraditionalSingleton-饿汉方式
	 * @throws Exception
	 */
	private static void attackTraditionalSingleton() throws Exception{
		Class<TraditionalSingleton> class_a=TraditionalSingleton.class;
		Constructor<TraditionalSingleton> constructor_a=class_a.getDeclaredConstructor();
		constructor_a.setAccessible(true);
		TraditionalSingleton obj=constructor_a.newInstance();
		obj=TraditionalSingleton.getInstance();
	}
	/**
	 * 2-LazySingleton-懒汉方式
	 * @throws Exception
	 */
	private static void attackLazySingleton() throws Exception{
		Class<LazySingleton> class_a=LazySingleton.class;
		Constructor<LazySingleton> constructor_a=class_a.getDeclaredConstructor();
		constructor_a.setAccessible(true);
		LazySingleton obj=constructor_a.newInstance();
		obj=LazySingleton.getInstance();
	}
	/**
	 * 3-StaticInnerClassSingleton-静态内部类方式
	 * @throws Exception
	 */
	private static void attackStaticInnerClassSingleton() throws Exception{
		Class<StaticInnerClassSingleton> class_a=StaticInnerClassSingleton.class;
		Constructor<StaticInnerClassSingleton> constructor_a=class_a.getDeclaredConstructor();
		constructor_a.setAccessible(true);
		StaticInnerClassSingleton obj=constructor_a.newInstance();
		obj=StaticInnerClassSingleton.getInstance();
	}
	/**
	 * 4-DoubleCheckSingleton-双重校验锁方式
	 * @throws Exception
	 */
	private static void attackDoubleCheckSingleton() throws Exception{
		Class<DoubleCheckSingleton> class_a=DoubleCheckSingleton.class;
		Constructor<DoubleCheckSingleton> constructor_a=class_a.getDeclaredConstructor();
		constructor_a.setAccessible(true);
		DoubleCheckSingleton obj=constructor_a.newInstance();
		obj=DoubleCheckSingleton.getInstance();
	}
	/**
	 * 5-EnumSingleton-枚举方式
*Java的单例模式在Effective Java里面有权威的做法,就是用Enum。这是最完美的做法。它
*利用Java自身语言机制保证了内存中只有一份拷贝,同时它也让那些想通过Reflection + *setAccessable()的攻击单例的做法失败。
	 * @throws Exception
	 */
	private static void attackEnumSingleton() throws Exception{
		Class<EnumSingleton> class_a=EnumSingleton.class;
		Constructor<EnumSingleton> constructor_a=class_a.getDeclaredConstructor();
		constructor_a.setAccessible(true);
		EnumSingleton obj=constructor_a.newInstance();
		obj=EnumSingleton.INSTANCE;
	}
	public static void main(String[] args) throws Exception {
		attackTraditionalSingleton();//输出2次
		attackLazySingleton();//输出2次
		attackStaticInnerClassSingleton();//输出2次
		attackDoubleCheckSingleton();//输出2次
		attackEnumSingleton();//抛出异常:NoSuchMethodException
	}
}

输出结果console截图:


  • 大小: 14.3 KB
分享到:
评论

相关推荐

    java单例设计模式-饿汉式-懒汉式 (2).pdf

    Java中的单例设计模式是一种确保一个类只有一个实例并提供一个全局访问点的设计模式。它主要分为两种实现方式:饿汉式(Eager Initialization)和懒汉式(Lazy Initialization)。 ### 1. 饿汉式单例设计模式 饿汉...

    java单例设计模式-饿汉式-懒汉式[归纳].pdf

    总结,Java单例设计模式通过饿汉式和懒汉式实现,饿汉式在类加载时创建实例,适用于不需要延迟加载的情况;懒汉式在首次调用时创建,但在多线程环境下需要额外的同步措施以保证线程安全。在实际应用中,应根据具体...

    java单例设计模式-饿汉式-懒汉式.docx

    Java中的单例设计模式是一种常用的设计模式,它确保一个类只有一个实例,并且提供一个全局访问点。单例模式常用于管理资源,如配置信息的读取,或者在系统中需要一个全局协调者的情况下。 单例设计模式有两大关键点...

    java单例设计模式-饿汉式-懒汉式.pdf

    Java 单例设计模式主要分为两种实现方式:饿汉式和懒汉式。这两种方式都是为了确保一个类只有一个实例,并且提供全局访问点。 **饿汉式**: 饿汉式单例在类加载时就完成了实例化,因此是静态常量的方式,确保了线程...

    java单例设计模式-饿汉式-懒汉式[参照].pdf

    在Java中,单例模式有两种常见的实现方式:饿汉式和懒汉式。 1. **饿汉式**: 饿汉式单例在类加载时即创建了实例,因此它是线程安全的。这种方式保证了类加载后就立即初始化单例对象,避免了多线程环境下的同步...

    java单例设计模式-饿汉式-懒汉式 (2).docx

    这种模式在Java中有两种常见的实现方式:饿汉式和懒汉式。 1. 饿汉式单例: 饿汉式单例在类加载时就完成了初始化,因此类加载时就创建了单例对象。这种方式称为预先加载方式,它保证了线程安全,因为对象的创建在类...

    java 单例模式(懒汉式与饿汉式).docx

    java 单例模式(懒汉式与饿汉式) java 单例模式是一种常用的软件设计模式。在它的可信结构中只包含一个被实例化单例的特别类。通过单例设计模式可以把整系统中的一个类只有一个实例。单例设计模式又分为两种方式,...

    java 单例模式(懒汉式与饿汉式)

    Java 单例模式(懒汉式与饿汉式) Java 单例模式是一种常用的软件设计模式,在它的可信结构中只包含一个被实例化单例的特殊类。通过单例设计模式可以把整系统中的一个类只有一个实例。单例设计模式又分为两种方式,...

    单例模式(饿汉模式、懒汉模式、DCL单例模式、枚举)

    本文将详细讨论四种常见的单例实现方式:饿汉模式、懒汉模式、双重检查锁定(DCL)单例模式以及枚举单例。 1. **饿汉模式**: 饿汉模式是在类加载时就完成了实例化,避免了线程同步问题。这种方式简单且安全,但...

    java-单例模式几种写法

    单例模式是软件设计模式中的一种,用于控制类的实例化过程,确保一个类只有一个实例,并提供全局访问点。在Java中,实现单例模式有多种方法,每种方法都有其特点和适用场景。以下是对这六种常见单例模式实现方式的...

    Java设计模式之单例模式的七种写法

    在 Java 中,单例模式的写法有好几种,主要有懒汉式单例、饿汉式单例、登记式单例等。 懒汉式单例是一种常见的单例模式实现方式,它在第一次调用的时候实例化自己。下面是懒汉式单例的四种写法: 1、基本懒汉式...

    java 单例模式(饿汉模式与懒汉模式).docx

    Java 单例模式是一种设计模式,它的核心思想是确保一个类只有一个实例,并提供一个全局访问点。这种模式在资源管理、缓存、日志记录等场景中广泛应用,例如在 JDBC 连接数据库时,通常会采用单例模式来管理数据库...

    Java单例模式实现静态内部类方法示例

    在Java中,单例模式可以通过多种方式实现,包括懒汉式、饿汉式、双重检查锁定和静态内部类方法等。今天,我们主要介绍了Java单例模式实现静态内部类方法示例,涉及构造函数私有化等相关内容。 单例模式的定义 单例...

    单例模式----数据库连接池管理类的应用

    单例模式是一种常用的软件设计模式,它保证一个类只有一个实例,并提供一个全局访问点。在IT行业中,尤其是在处理资源密集型任务如数据库连接时,单例模式被广泛应用。数据库连接池就是这种应用的一个典型例子。 ...

    Java 单例模式 工具类

    Java中的单例模式是一种常用的软件设计模式,它保证一个类只有一个实例,并提供全局访问点。在Java编程中,单例模式常用于控制资源的访问,比如数据库连接池、线程池或者日志对象等。本篇文章将深入探讨如何在Java中...

    java单例模式实例

    单例模式是软件设计模式中的一种经典模式,用于确保一个类只有一个实例,并提供一个全局访问点。在Java中,有多种实现单例模式的方法,每种都有其特点和适用场景。接下来,我们将深入探讨这些实现方式。 首先,我们...

    设计模式——单例模式

    实现单例模式主要有两种方式:饿汉式和懒汉式。 ### **饿汉式单例模式** 饿汉式单例模式在类加载时就完成了实例化,因此也称为静态初始化。这种方式保证了线程安全,但可能会造成不必要的资源浪费,因为即使未使用...

    java设计模式----单例模式

    单例模式分为两种主要实现方式:饿汉模式和懒汉模式。 1. 饿汉模式(Eager Initialization): 饿汉模式的特点是在类加载时就完成了实例化,因此它是线程安全的。在给出的例子中,`SingleTon`类的静态私有实例`st`...

    单例模式,懒汉模式和饿汉模式

    单例模式是软件设计模式中的一种,用于控制类的实例化过程,确保一个类在整个程序运行期间只有一个实例存在。这种模式常用于管理共享资源,如数据库连接、缓存或者线程池等,以减少资源消耗并提高性能。单例模式有...

Global site tag (gtag.js) - Google Analytics