第一种(懒汉,线程不安全):
-
public
class
Singleton {
-
private
static
Singleton instance;
-
-
public
static
Singleton getInstance() {
-
if
(instance ==
null
) {
-
instance = new
Singleton();
-
}
-
return
instance;
-
}
-
}
这种写法lazy loading很明显,但是致命的是在多线程不能正常工作。
第二种(懒汉,线程安全):
-
public
class
Singleton {
-
private
static
Singleton instance;
-
-
public
static
synchronized
Singleton getInstance() {
-
if
(instance ==
null
) {
-
instance = new
Singleton();
-
}
-
return
instance;
-
}
-
}
这种写法能够在多线程中很好的工作,而且看起来它也具备很好的lazy loading,但是,遗憾的是,效率很低,99%情况下不需要同步。
第三种(饿汉):
-
public
class
Singleton {
-
private
static
Singleton instance =
new
Singleton();
-
-
public
static
Singleton getInstance() {
-
return
instance;
-
}
-
}
这种方式基于
classloder
机制避免了多线程的同步问题,不过,
instance
在类装载时就实例化,虽然导致类装载的原因有很多种,在单例模式中大多数都是调用
getInstance
方法,
但是也不能确定有其他的方式(或者其他的静态方法)导致类装载,这时候初始化
instance
显然没有达到
lazy loading
的效果。
第四种(饿
汉,变种):
-
public
class
Singleton {
-
private
Singleton instance =
null
;
-
static
{
-
instance = new
Singleton();
-
}
-
-
public
static
Singleton getInstance() {
-
return
this
.instance;
-
}
-
}
表面上看起来差别挺大,其实更第三种方式差不多,都是在类初始化即实例化instance。
第五种(静态内部类):
-
public
class
Singleton {
-
private
static
class
SingletonHolder {
-
private
static
final
Singleton INSTANCE =
new
Singleton();
-
}
-
-
public
static
final
Singleton getInstance() {
-
return
SingletonHolder.INSTANCE;
-
}
-
}
这种方式同样利用了
classloder
的机制来保证初始化
instance
时只有一个线程,它跟第三种和第四种方式不同的是(很细微的差别):第三种和第四种方式是只要
Singleton
类被装载了,那么
instance
就会被实例化(没有达到
lazy loading
效果),而这种方式是
Singleton
类被装载了,
instance
不一定被初始化。因为
SingletonHolder
类没有被主动使用,只有显示通过调用
getInstance
方法时,才会显示装载
SingletonHolder
类,从而实例化
instance
。想象一下,如果实例化
instance
很消耗资源,我想让他延迟加载,另外一方面,我不希望在
Singleton
类加载时就实例化,因为我不能确保
Singleton
类还可能在其他的地方被主动使用从而被加载,那么这个时候实例化
instance
显然是不合适的。这个时候,这种方式相比第三和第四种方式就显得很合理。
第六种(枚举):
-
public
enum
Singleton {
-
INSTANCE;
-
public
void
whateverMethod() {
-
}
-
}
这种方式是Effective Java作者Josh Bloch
提倡的方式,它不仅能避免多线程同步问题,而且还能防止反序列化重新创建新的对象,可谓是很坚强的壁垒啊,不过,个人认为由于1.5中才加入enum特
性,用这种方式写不免让人感觉生疏,在实际工作中,我也很少看见有人这么写过。
第七种(双重校验锁):
-
public
class
Singleton {
-
private
volatile
static
Singleton singleton;
-
-
public
static
Singleton getSingleton() {
-
if
(singleton ==
null
) {
-
synchronized
(Singleton.
class
) {
-
if
(singleton ==
null
) {
-
singleton = new
Singleton();
-
}
-
}
-
}
-
return
singleton;
-
}
-
}
这个是第二种方式的升级版,俗称双重检查锁定,详细介绍请查看:http://www.ibm.com/developerworks/cn/java/j-dcl.html
在JDK1.5之后,双重检查锁定才能够正常达到单例效果。
总结
有两个问题需要注意:
1.如果单例由不同的类装载器装入,那便有可能存在多个单例类的实例。假定不是远端存取,例如一些servlet
容器对每个servlet
使用完全不同的类装载器,这样的话如果有两个servlet
访问一个单例类,它们就都会有各自的实例。
2.如果Singleton
实现了java.io.Serializable
接口,那么这个类的实例就可能被序列化和复原。不管怎样,如果你序列化一个单例类的对象,接下来复原多个那个对象,那你就会有多个单例类的实例。
对第一个问题修复的办法是:
-
private
static
Class getClass(String classname)
-
throws
ClassNotFoundException {
-
ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
-
-
if
(classLoader ==
null
)
-
classLoader = Singleton.class
.getClassLoader();
-
-
return
(classLoader.loadClass(classname));
-
}
-
}
对第二个问题修复的办法是:
-
public
class
Singleton
implements
java.io.Serializable {
-
public
static
Singleton INSTANCE =
new
Singleton();
-
-
protected
Singleton() {
-
-
}
-
private
Object readResolve() {
-
return
INSTANCE;
-
}
-
}
对我来说,我比较喜欢第三种和第五种方式,简单易懂,而且在JVM层实现了线程安全(如果不是多个类加载器环境),一般的情况下,我会使用第三种方
式,只有在要明确实现lazy
loading效果时才会使用第五种方式,另外,如果涉及到反序列化创建对象时我会试着使用枚举的方式来实现单例,不过,我一直会保证我的程序是线程安全
的,而且我永远不会使用第一种和第二种方式,如果有其他特殊的需求,我可能会使用第七种方式,毕竟,JDK1.5已经没有双重检查锁定的问题了。
========================================================================
superheizai
同学总结的很到位:
不过一般来说,第一种不算单例,第四种和第三种就是一种,如果算的话,第五种也可以分开写了。所以说,一般单例都是五种写法。懒汉,恶汉,双重校验锁,枚举和静态内部类。
我很高兴有这样的读者,一起共勉。
鉴于很多人说我没有写私有构造方法,我这里声明一下,不是我忘了写,是我故意嫌麻烦省略掉了,我以为大家都懂得
分享到:
相关推荐
单例模式是软件设计模式中的一种,它的主要目的是确保一个类只有一个实例,并提供一个全局访问点。这种模式在很多场景下都非常有用,比如控制资源的唯一性、管理共享配置或者创建昂贵的对象时避免频繁创建销毁。 ...
Java设计模式之单例模式的七种写法 单例模式是一种常见的设计模式,它确保某个类只有一个实例,而且自行实例化并向整个系统提供这个实例。在计算机系统中,线程池、缓存、日志对象、对话框、打印机的驱动程序对象常...
### 单例模式应用场景 #### 一、概述 ...综上所述,虽然单例模式被认为是一种相对简单的实现技巧,但在实际开发中却有着广泛的应用前景。合理地使用单例模式,可以显著提升软件系统的稳定性和性能。
单例模式是软件设计模式中的一种,其主要目的是控制类的实例化过程,确保一个类在整个应用程序中只有一个实例存在。这种模式在很多场景下都非常有用,例如管理共享资源、配置对象或者全局日志服务等。下面我们将深入...
单例模式是软件设计模式中的一种经典模式,用于确保一个类只有一个实例,并提供一个全局访问点。这种模式在很多场景下都非常有用,比如控制数据库连接、管理缓存或者全局配置等。下面我们将详细探讨单例模式的七种...
在软件设计模式中,工厂模式和单例模式是两种非常基础且重要的模式,它们都是用于解决对象创建问题,但有着不同的设计理念和应用场景。本篇文章将深入探讨这两种模式,并结合具体的代码示例`myFactoryDemo`进行讲解...
单例模式是一种广泛应用于软件设计中的创建型设计模式,它的核心思想是确保一个类只有一个实例,并提供一个全局访问点。这样做的好处在于控制共享资源的访问,比如线程安全的数据库连接池或者配置管理等。 在Java中...
单例模式是软件设计模式中的一种经典模式,它在Java编程中被广泛使用。这个模式的主要目的是确保一个类只有一个实例,并提供一个全局访问点。这样做的好处包括资源管理(如数据库连接)、性能优化(如缓存服务)以及...
Java的单例模式是一种常用的软件设计模式,它保证一个类只有一个实例,并提供全局访问点。在Android开发中,单例模式的应用尤其广泛,因为它能够有效管理资源,减少内存开销,提高性能。以下是对单例模式在Android中...
在《设计模式》第二版中,刘伟老师深入讲解了单例模式,本压缩包中的"单例模式习题6"和"单例模式习题7"可能是书中的课后练习,旨在帮助读者更好地理解和运用单例模式。 单例模式的核心思想是控制类的实例化过程,...
单例模式是软件设计模式中的一种经典模式,它主要用于控制类的实例化过程,确保一个类在整个应用程序中只有一个实例存在。这种模式在Java编程中非常常见,特别是在需要频繁实例化然后销毁的对象,或者需要共享资源的...
单例模式是软件设计模式中的一种经典模式,它限制了类的实例化过程,确保一个类在整个系统中只有一个实例存在。这种模式在处理全局资源、线程共享对象以及频繁使用的对象时尤其有用,因为它可以避免频繁创建和销毁...
单例模式是软件设计模式中的一种,它的主要目的是确保一个类只有一个实例,并提供一个全局访问点。这种模式在很多场景下都非常有用,比如控制服务的唯一性、避免资源浪费等。接下来,我们将深入探讨几种常见的单例...
### Java单例模式开发的七种写法 #### 概述 单例模式是一种常用的软件设计模式,其目的是确保一个类仅有一个实例,并提供一个全局访问点。在Java编程语言中,实现单例模式的方法有很多种,不同的实现方式具有不同的...
单例模式是软件设计模式中的一种经典模式,它限制一个类只能有一个实例存在。在Java中,单例模式常用于管理共享资源,如数据库连接池、线程池或配置信息等,以确保这些资源在整个应用生命周期中只被创建一次。在多...
单例模式是软件设计模式中的一种经典模式,它在软件体系结构中扮演着重要的角色。这个模式的主要目的是确保一个类在整个应用程序中只有一个实例,并提供一个全局访问点来获取这个唯一的实例。这种设计模式广泛应用于...
7. **SingleTest.h**和`SingleTest.ui`:这分别是测试类的头文件和UI设计文件,可能用于创建一个简单的界面来展示单例模式的工作情况。 8. **SingleTest.qrc**:这是一个资源文件,用于将非代码资源(如图像、音频...
单例模式是软件设计模式中的一种,它保证一个类只有一个实例,并提供一个全局访问点。这种模式在资源管理、配置对象、线程池等场景中非常常见,因为这些场景往往需要确保系统中只有一个实例来协调整个系统的状态。...