`
javandroid
  • 浏览: 25574 次
  • 性别: Icon_minigender_1
文章分类
社区版块
存档分类
最新评论

设计模式-单例模式(Singleton)

 
阅读更多

定义: Singleton模式主要作用是保证在Java应用程序中,一个类Class只有一个实例存在。

应用场景:比如建立目录或数据库连接都需要这样的单例。

实现思路:
1.私有的构造器,以防止外部new出多个实例
2.在内部提供一个单实例
3.向外部暴露一个获取该单实例的方法

4.同步控制,防止生成多个实例


实现单例模式常用的几种方式:饿汉式、懒汉式、懒汉双检索式、内部类实现式、枚举实现式等

一、饿汉式

1.饿汉式(线程安全)

class Singleton {	
	private static Singleton instance = new Singleton();

	private Singleton() {}

	public static Singleton getInstance() { 
		return instance;
	}
}

类被加载时就对instance进行实例化,单实例就被创建了。养兵千日用兵一时,不管以后用不用的着,先创建再说,以空间换时间。

优点:这种写法简单,类装载时完成实例化。避免了线程安全问题。

缺点:没达到懒加载效果。如果该实例一直没使用到,但它却一直占用内存空间。

2.饿汉式变体(线程安全)

class Singleton {	
	private static Singleton instance;

	static{
		instance = new Singleton();
	}
	
	private Singleton() {}

	public static Singleton getInstance() { 
		return instance;
	}
}

同上

二、懒汉式

1.懒汉式(线程安全)

class Singleton {	
	private static Singleton instance;

	private Singleton() {}

	public static synchronized Singleton getInstance() { 
		if (instance == null) {
			instance = new Singleton();
		}
		return instance;
	}
}

缺点:每次获取实例都要进行同步,其实一旦创建了实例,再获取时只需要返回而无需进行同步。所以效率低。

2.不安全的改进(线程不安全)

改进。改成同步代码块实现,缩小临界区。

public class Singleton {

    private static Singleton singleton;

    private Singleton() {}

    public static Singleton getInstance() {
        if (singleton == null) {
            synchronized (Singleton.class) {
                singleton = new Singleton();
            }
        }
        return singleton;
    }
}

但是出现了线程安全问题。

假如一个线程进入了if (singleton == null)判断语句块,还未来得及往下执行,另一个线程也通过了这个判断语句,这时便会产生多个实例,所以这种方式是非线程安全的。

3.不安全的双检索方式(非线程安全)

继续改进。在同步时,再进行一次判空操作。这种方式叫做"双重检查锁",可以简称"双检锁"。

public class Singleton {
     
    private static Singleton singleton;

    private Singleton() {}

    public static Singleton getInstance() {
        if (singleton == null) {
            synchronized (Singleton.class) {
                if (singleton == null) {
                    singleton = new Singleton();
                }
            }
        }
        return singleton;
    }
}

这种方式好像能解决问题,但是实际上这样做依然无济于事。原因很多,指令的重排序,内存不可见等。

4.安全的双检索方式(线程安全)

最简单的解决上面双检索不安全的方式就是使用volatile。
public class Singleton {
    //用volatile修饰
    private static volatile Singleton singleton;

    private Singleton() {}

    public static Singleton getInstance() {
        if (singleton == null) {
            synchronized (Singleton.class) {
                if (singleton == null) {
                    singleton = new Singleton();
                }
            }
        }
        return singleton;
    }
}

该版本的双检锁机制的实现使用关键字volatile,它的意思是:被volatile修饰的变量的值,将不会被本地线程缓存,所有对该变量的读写都是直接操作共享内存,从而确保多个线程能正确的处理该变量。

注意:在java1.4及以前版本中,很多JVM对于volatile关键字的实现的问题,会导致“双重检查加锁”的失败,因此“双重检查加锁”机制只能用在java5及以上的版本。

由于volatile关键字可能会屏蔽掉虚拟机中一些必要的代码优化,所以运行效率可能并不高。而且有其它解决方案,因此该实现版本并不推荐

关于使用双检索时不加volatile可能导致的线程不安全的分析,请参考

The "Double-Checked Locking is Broken" Declaration和对应的翻译可以不要再使用Double-Checked Locking了

三、内部类实现方式

public class Singleton {

    private Singleton() {}
    
    //内部类
    private static class SingletonInstance {
        private static final Singleton INSTANCE = new Singleton();
    }

    public static Singleton getInstance() {
        return SingletonInstance.INSTANCE;
    }
}
这种方式跟饿汉式方式采用的机制类似,但又有不同。两者都是采用了类装载的机制来保证初始化实例时只有一个线程。不同的地方在饿汉式方式是只要Singleton类被装载就会实例化,没有Lazy-Loading的作用,而静态内部类方式在Singleton类被装载时并不会立即实例化,而是在需要实例化时,调用getInstance方法,才会装载SingletonInstance类,从而完成Singleton的实例化。
类的静态属性只会在第一次加载类的时候初始化,所以在这里,JVM帮助我们保证了线程的安全性,在类进行初始化时,别的线程是无法进入的。

优点:延迟加载,线程安全,效率高。

四、枚举实现方式

public enum Singleton {
    INSTANCE;
}
借助JDK1.5中添加的枚举来实现单例模式。不仅能避免多线程同步问题,而且还能防止反序列化重新创建新的对象。可能是因为枚举在JDK1.5中才添加,所以在实际项目开发中,很少见人这么写过。
优点:
系统内存中该类只存在一个对象,节省了系统资源,对于一些需要频繁创建销毁的对象,使用单例模式可以提高系统性能。
缺点:
当想实例化一个单例类的时候,必须要记住使用相应的获取对象的方法,而不是使用new,可能会给其他开发人员造成困扰,特别是看不到源码的时候。


参考博客:

《JAVA与模式》之单例模式

单例模式的线程安全问题:探索设计模式之六——单例模式

分享到:
评论

相关推荐

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

    在Java编程语言中,设计模式是一种经过验证的解决常见软件设计问题的最佳实践。单例模式是其中最常用的一种,它的核心思想是确保一个类只有一个实例,并提供全局访问点。单例模式的应用场景包括:控制资源的访问、...

    ava常用设计模式-单例模式

    单例模式(Singleton Pattern)是 Java 中最简单的设计模式之一,有以下特点: 1. 单例类只能有一个实例。 2. 单例类必须自己创建自己的唯一实例。 3. 单例类必须给所有其他对象提供这一实例。 单例模式有多种实现...

    Java设计模式-单例模式详解

    Java设计模式-单例模式详解 单例模式是 Java 设计模式中的一种常用的设计模式,旨在保证一个类仅有一个实例,并提供一个访问它的全局访问点。单例模式的目的是为了保证在一个进程中,某个类有且仅有一个实例。 ...

    常见设计模式-单例模式

    设计模式-单例模式 单例模式是一种常见的设计模式,它的主要作用是确保在内存中仅创建一次对象,并提供一个全局访问点。单例模式有两种类型:饿汉类型和懒汉类型。 饿汉类型的单例模式是在类加载时创建对象,例如...

    设计模式-单例模式

    **设计模式——单例模式** 单例模式是一种广泛应用于软件设计中的创建型设计模式,它的核心思想是确保一个类只有一个实例,并提供一个全局访问点。这样做的好处在于控制共享资源的访问,比如线程安全的数据库连接池...

    设计模式-单例设计模式

    单例模式是一种常见的创建型设计模式,其核心思想在于确保一个类仅有一个实例存在,并且该实例由该类自行创建,随后向整个系统提供这一唯一实例。 #### 类图解析 在设计模式中,类图是一种直观展示类结构及关系的...

    PHP5设计模式-单例模式

    单例模式是一种常用的设计模式,它在软件工程中扮演着控制类实例化过程的角色,确保一个类只有一个实例,并提供全局访问点。这种模式在PHP中尤其常见,特别是在需要频繁实例化然后销毁的对象,或者需要控制共享资源...

    java设计模式-单例.pdf

    ### Java设计模式——单例模式详解 #### 一、单例模式概述 单例模式是设计模式中的一个重要组成部分,属于创建型模式之一。其主要作用是确保某个类仅有一个实例存在,并提供一个全局访问该实例的方法。这在很多场景...

    c++设计模式-单例模式

    **单例模式**是软件设计模式中的一种,属于创建型模式。在C++中,单例模式确保一个类只有一个实例,并提供一个全局访问点。这种模式通常用于管理共享资源,如数据库连接、日志系统或者配置文件等。单例模式的主要...

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

    单例模式是软件设计模式中的一种经典模式,它在Java编程中被广泛使用。这个模式的主要目的是确保一个类只有一个实例,并提供一个全局访问点。这样做的好处在于可以控制实例的数量,减少资源消耗,同时便于协调整个...

    设计模式-单例模式(讲解及其实现代码)

    单例模式是软件设计模式中的一种,它保证一个类只有一个实例,并提供全局访问点。这种模式在许多场景下非常有用,比如控制共享资源、管理系统级别的对象,如数据库连接池或者线程池等。单例模式的核心在于限制类的...

    大话设计模式--Singleton(单例模式)

    单例模式是软件设计模式中的一种经典模式,它在许多场景下被广泛使用,尤其是在需要全局唯一实例的情况下。本文将深入探讨单例模式的概念、作用、实现方式以及其在实际编程中的应用。 单例模式的核心思想是确保一个...

    通过C++实现设计模式-单例模式.rar

    压缩包代码是一个简单的C++单例模式实现。 在这个例子中,Singleton 类有一个私有的构造函数和一个私有的析构函数,这防止了外部代码直接创建或销毁实例。getInstance 方法是一个静态方法,它返回一个指向 Singleton...

    Java-设计模式-单例模式-实现源码(简单实现、双重检查锁、静态内部类、枚举类)

    单例模式是软件设计模式中的一种经典模式,其主要目的是保证一个类只有一个实例,并提供一个全局访问点。在Java中,有多种实现单例模式的方法,包括简单实现、双重检查锁定(Double-Checked Locking)、静态内部类和...

    Java面向对象(高级)- 单例(Singleton)设计模式

    总的来说,单例模式是一种强大的设计模式,它在许多场合下都能帮助我们更好地组织代码,控制资源,提高程序的可维护性和可扩展性。在实际开发中,应根据具体需求选择合适的设计模式,以达到最佳效果。

    优秀的设计模式示例-单例模式

    单例模式是软件设计模式中的一种经典模式,它确保一个类只有一个实例,并提供一个全局访问点。这种模式在很多场景下都非常有用,比如控制共享资源、管理系统级别的服务或者简化对象间的交互。以下是对单例模式的详细...

    设计模式ForiOS-02-单例模式.pdf

    单例模式是一种软件设计模式,它的核心思想是确保一个类在整个应用程序中只有一个实例,并提供一个全局访问点来获取这个实例。这种模式在iOS开发中非常常见,因为许多系统类如UIApplication、NSFileManager等都采用...

    IOS应用源码Demo-单例模式-毕设学习.zip

    在iOS应用开发中,单例模式是一种常用的编程设计模式,它确保了类只有一个实例,并提供一个全局访问点。这个模式在iOS开发中的应用非常广泛,特别是在处理系统资源、网络请求、数据存储等需要全局共享的对象时。在...

    C#设计模式学习与演变过程-2-单例模式

    单例模式是软件设计模式中的一种经典模式,它旨在确保一个类只有一个实例,并提供一个全局访问点。在C#编程中,单例模式的应用广泛,特别是在需要频繁实例化然后销毁的对象,或者创建对象需要消耗大量资源的情况下,...

Global site tag (gtag.js) - Google Analytics