Android中的设计模式之单例模式

简介: 参考《设计模式:可复用面向对象软件的基础 》3.5 Singleton 单件--对象创建型模式《设计模式解析》(第二版)第21章 Singleton模式和Double-Checked Locking模式《Android源码设计模式解析与实战》第2章 单例模式意图保证一个类仅有一个实例,并提供一个访问它的全局访问点。

参考

  • 《设计模式:可复用面向对象软件的基础 》3.5 Singleton 单件--对象创建型模式
  • 《设计模式解析》(第二版)第21章 Singleton模式和Double-Checked Locking模式
  • 《Android源码设计模式解析与实战》第2章 单例模式

意图

保证一个类仅有一个实例,并提供一个访问它的全局访问点。

动机

对一些类来说,只有一个类实例是很重要的,比如一个公司CEO只有一位,一个帝国只有一个皇帝。
单例模式是应用最广的模式之一,单例对象的类必须保证只有一个实例存在。如在一个应用中,应该只有一个ImageLoader实例,这个ImageLoader中又含有线程池,缓存系统,网络请求等,很消耗资源,因此,没有理由让它构造多个实例。

保持只有一个单例,好的办法是让类自身负责保存它的唯一实例。

适用性

  • 当类只能有一个实例而且客户可以从一个众所周知的访问点访问它。
  • 当这个唯一实例应该是通过子类化可扩展的,并且客户应该无需更改代码就能使用一个扩展的实例。

定义

确保一个类只有一个实例,而且自行实例化并向整个系统提供这个实例

角色

  • Client 高层客户端
  • Singleton 单例类
Singleton

Java实现单例模式的关键点

  • 构造函数不对外开放,一般为private;
  • 通过一个静态方法或者枚举返回单例类对象;
  • 确保单例类的对象有且只有一个,尤其是在多线程环境下;
  • 确保单例类对象在反序列化时不会重新构建对象;

实现方式

饿汉模式

public class Singleton {
    private void Singleton(){}
    private static Singleton instance =new Singleton();
    public static Singleton getInstance(){
        return instance;
    }
}

加载类的时候就构造了单例,有个问题是万一客户端一直都不用这个单例呢,岂不是浪费资源

懒汉模式

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

这里getInstance()方法添加了synchronized关键字,也就是getInstance()是一个同步方法,目的是保证在多线程情况下单例对象唯一的手段。这里会有个问题,即使instance已经被初始化,每次调用getInstance()方法都会进行同步,这样会消耗不必要的资源。

Double Check Lock(DCL)实现单例

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

因为getInstance()可能同时会有多个客户端调用,DCL方式实现单例模式的优点是既能在需要时才初始化单例,又能保证线程安全,且单例对象初始化调用后调用getInstance不进行同步锁。

  • 第一次检查单例有没有实例化, getInstance()并没有同步,所以不会太耗时,如果为空则进入第二次检查,否则返回实例
  • 第二次检查就是构造实例了,这次必须同步,就是在同步的情况下,判断实例为不为空,为空就构造。

这种实现方式很流行,很多框架都用这种

静态内部类单例模式

public class Singleton {
    private void Singleton(){}
    
    public  static Singleton getInstance(){
        return SingletonHolder.sInstance;
    }
    /**
     * 静态内部类
     */
    private static class SingletonHolder{
        private static final Singleton sInstance=new Singleton();
    }
}

DCL虽然在一定程度上解决了资源消耗,多余的同步,线程安全等问题,但是,他还是在某些情况下出项失效的问题。

静态内部类单例模式,当第一次加载Singleton类时并不会初始化sInstance,只有在第一次调用Singleton的getInstance方法时才会导致sInstance被初始化。因此,第一次调用getInstance方法会导致虚拟机加载SinglentonHolder类,这种方式不仅能确保线程安全,也能够保证单例对象的唯一性,同时也延迟了单例的实例化,所以这个是推荐使用的单例模式实现方式。

枚举单例

public enum SingletonEnum {
    INSTANCE;
    public void dosomeThing(){
        System.out.println("do sth.");
    }

}

《Effective Java》推荐的用法,用枚举

使用容器实现单例模式

public class SingletonManager {
    private static Map<String,Object> objMap=new HashMap<String,Object>();
    
    private SingletonManager(){}
    public static void registerService(String key,Object instance){
        if (!objMap.containsKey(key)) {
            objMap.put(key, instance);
        }
    }
    public static Object getService(String key){
        return objMap.get(key);
    }
}

在程序的初始,将多种单例类型注入到一个统一的管理类中,在使用时根据key获取对象对应类型的对象。

优点

  • 减少内存开支,特别是在一个对象频繁的创建和销毁时。
  • 单例模式可以避免对资源的多重占用
  • 单例模式可以在系统中设置全局的访问点,优化和共享资源访问

缺点

  • 单例模式一般没有接口,扩展很困难
  • 单例对象如果持有Context,容易发生内容泄露,最少congtext最好是Application级别的

应用例子1 Android开发封装的OkHttp3 Client单例

// todo 这个很常见了,后面贴个代码

应用例子2 Android中的LayoutInflater

// todo 待研究,后面更

目录
相关文章
|
1月前
|
设计模式 存储 SQL
PHP中的设计模式:单例模式的探索
在PHP开发中,单例模式是一种常用的设计模式,它确保一个类只有一个实例,并提供一个全局访问点。本文将通过一个简单的例子,逐步引导你理解如何在PHP中实现和利用单例模式,以及它在实际项目中的应用价值。
|
3月前
|
设计模式 缓存 安全
Java设计模式的单例模式应用场景
Java设计模式的单例模式应用场景
42 4
|
3月前
|
设计模式 SQL 安全
【设计模式】第二篇:单例模式的几种实现And反射对其的破坏
一个普通实例化,一个反射实例化 但是我们如果通过反射的方式进行实例化类,会有什么问题呢? public static void main(String[] args) throws Exception { Lazy1 lazy1 = getLazy1();
29 5
|
1月前
|
设计模式 存储 负载均衡
【五】设计模式~~~创建型模式~~~单例模式(Java)
文章详细介绍了单例模式(Singleton Pattern),这是一种确保一个类只有一个实例,并提供全局访问点的设计模式。文中通过Windows任务管理器的例子阐述了单例模式的动机,解释了如何通过私有构造函数、静态私有成员变量和公有静态方法实现单例模式。接着,通过负载均衡器的案例展示了单例模式的应用,并讨论了单例模式的优点、缺点以及适用场景。最后,文章还探讨了饿汉式和懒汉式单例的实现方式及其比较。
【五】设计模式~~~创建型模式~~~单例模式(Java)
|
1月前
|
设计模式 安全 程序员
C#设计模式之单例模式
C#设计模式之单例模式
40 3
|
29天前
|
设计模式 SQL 缓存
Java编程中的设计模式:单例模式的深入理解与应用
【8月更文挑战第22天】 在Java的世界里,设计模式是构建可维护、可扩展和灵活的软件系统的基石。本文将深入浅出地探讨单例模式这一经典设计模式,揭示其背后的哲学思想,并通过实例演示如何在Java项目中有效运用。无论你是初学者还是资深开发者,这篇文章都将为你打开一扇洞悉软件设计深层逻辑的大门。
26 0
|
1月前
|
设计模式 存储 数据库连接
Python设计模式:巧用元类创建单例模式!
Python设计模式:巧用元类创建单例模式!
31 0
|
1月前
|
设计模式 安全 测试技术
[设计模式]创建型模式-单例模式
[设计模式]创建型模式-单例模式
|
2月前
|
设计模式 安全 C++
C++一分钟之-C++中的设计模式:单例模式
【7月更文挑战第13天】单例模式确保类只有一个实例,提供全局访问。C++中的实现涉及线程安全和生命周期管理。基础实现使用静态成员,但在多线程环境下可能导致多个实例。为解决此问题,采用双重检查锁定和`std::mutex`保证安全。使用`std::unique_ptr`管理生命周期,防止析构异常和内存泄漏。理解和正确应用单例模式能提升软件的效率与可维护性。
32 2
|
2月前
|
设计模式 安全 Java
Java面试题:设计模式如单例模式、工厂模式、观察者模式等在多线程环境下线程安全问题,Java内存模型定义了线程如何与内存交互,包括原子性、可见性、有序性,并发框架提供了更高层次的并发任务处理能力
Java面试题:设计模式如单例模式、工厂模式、观察者模式等在多线程环境下线程安全问题,Java内存模型定义了线程如何与内存交互,包括原子性、可见性、有序性,并发框架提供了更高层次的并发任务处理能力
60 1