Android设计模式之单例模式 Singleton

一.概述

单例模式是设计模式中最简单的一种,但是它没有设计模式中的那种各种对象之间的抽象关系,所以有人不认为它是一种模式,而是一种实现技巧.单例模式就像字面的意思一样,提供一个只能自己实例化的实例,并且提供了一个全局的访问点.要达到这几点要求就要满足三点:私有构造函数(防止被别人实例化),静态私有自身对象(用来提供实例),静态公有的getInstance方法(用来创建和获取实例对象).

优缺点:

单例只允许自己建立一个实例,不需要频繁创建和销毁,可以节省内存加快对象的访问速度.

但是单例没有抽象层和借口,不方便扩展.单例既提供工厂方法又提供业务方法,一定程度上违背了单一职责原则

二.单例实现

单例的实现有两个主流方式,分别是懒汉模式和饿汉模式,他们在实例化的时机和效率方面各有不同

1.懒汉模式

/**
 * Created by jesse on 15-6-28.
 */
public class Singleton {
    private static Singleton instance;
    private Singleton() {} //一定要有私有构造,要不谈何单例
    public static Singleton getInstance(){
        if (null == instance){
            instance = new Singleton();
        }
        return instance;
    }
}

懒汉模式在外部对象每次获取实例时都要先判断该实例是否被初始化,这点相比饿汉模式来说就会损失一些效率,但是会节省一些空间,因为什么时候用到该实例才会去初始化,如果一直用不到的话,在懒汉模式里面时不会构造该对象的.相当于用空间换时间.也就是延迟加载技术.还有一点需要注意的是一定要有私有构造,要不然外部对象还是可以实例化该对象,那还谈何单例.

这个时候很多小伙伴要说了,你这个懒汉单例有问题,有线程安全问题.当然这个方式是线程不安全的,当两个线程A和B并发要获取实例的时候,instance还没有被初始化,假设A先拿到时间片去初始化instance,当instance还没有初始化完成的时候,时间片让给了B,这时由于instance还没有被初始化完成那么instance还为空,所以B还会去把instance实例化一次,最终的结果就是instance被实例化了两次.这可能会造成很严重的后果,解决方式就是加锁(牺牲一些性能),使用饿汉模式.

给getInstance方法加锁,这种加锁方式可以解决线程安全问题,但是每次外部对象要获取实例的时候都要进行线程锁定,之后还要再判断instance是否被实例化了,这样在多线程高并发的情况下效率损失就很可观了.

public class Singleton {
    private static Singleton instance;
    private Singleton() {}
    public static synchronized Singleton getInstance(){//给getInstance方法加锁
        if (null == instance){
            instance = new Singleton();
        }
        return instance;
    }
}

后来又延伸出了一种双检锁实例的方式,这种方式只在第一次实例化的时候进行加锁,并在在加锁前后都会对是否实例化了进行判定.性能方面优于单检锁的形式.但是这种双检锁还会遇到一个问题,就是在不同的平台或不同的编译器下可能出现错误.主要是因为有些编译器优化了new Singleton()的过程,假设线程AB并发获取实例,在有些编译器下可能会出现A线程实例化instance的过程中,当系统已经给instance分配到内存,但是还没有初始化instance的成员变量的时候,时间片让给了B进程,这时候instance已经不为null了,B就直接拿着实例去操作成员变量,但是这个时候成员变量还没有被初始化,结果就可能crash了.当遇到这种编译器问题的时候就需要给instance变量加上volatile修饰,去除编译器优化的干扰.

public class Singleton {
    private static Singleton instance;
    //private static volatile Singleton instance; 遇到编译器问题时使用
    private Singleton() {}
    public static Singleton getInstance(){
        if (null == instance){
            synchronized(Singleton.class){
                if (null == instance)
                    instance = new Singleton();
            }

        }
        return instance;
    }
}

2.饿汉模式

饿汉模式是在系统运行起来,在装在类的时候就进行初始化的操作,外部对象使用的时候不需要做任何判断可以直接使用,从效率上来说是优于懒汉模式的.但是对比懒汉模式的延迟加载技术,不管系统用不用该实例,内存都会在最开始的时候创建出来.跟懒汉的时间换空间正好相反,饿汉是空间换时间.

public class Singleton {
    private static final Singleton instance = new Singleton();
    private Singleton() {}//一定要有私有构造,要不谈何单例
    public static Singleton getInstance(){
        return instance;
    }
}

三.总结

根据上面的分析可以很清楚的了解到:饿汉模式不用面对对线程并发的问题.并且从调用速度和响应时间来说饿汉模式要优于懒汉模式.但是从资源利用的角度来说懒汉模式要优于饿汉模式.

转载请注明出处: http://blog.csdn.net/l2show/article/details/46672061

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-17 16:58:19

Android设计模式之单例模式 Singleton的相关文章

设计模式之单例模式——Singleton

                    设计模式之单例模式--Singleton 设计意图: 保证类仅有一个实例,并且可以供应用程序全局使用.为了保证这一点,就需要这个类自己创建自己的对象,并且对外有公开的调用方法.而且,别的类不能实例化它,所以构造方法要设置为私有的. 单例模式的要点 一是某个类只能有一个实例: 二是它必须自行创建这个实例: 三是它必须自行向整个系统提供这个实例. 例如: 有一个"单例对象",而"客户甲"."客户乙" 和&quo

java设计模式之单例模式(Singleton)

Java设计模式之单例模式 单例模式是什么? 保证一个类仅有一个实例,并提供一个访问它的全局访问点. 单例模式如何来设计呢? 保证一个类只能有一个实例,那么我们不能无限制的new 来创建,因为我们知道,new一次就是一个新的对象,那么构造器只能私有化private -- 构造器私有化 构造器私有化了,问题又出现了,构造器私有化了,那么我们怎么来创建唯一的对象呢?-- 提供一个公有的方法/提供一个公有的静态属性 再分析一下,公有方法, 实例方法还是类方法呢?--公有的类方法(获取类实例) 依据以上

设计模式之——单例模式(Singleton)的常见应用场景(转):

单例模式(Singleton)也叫单态模式,是设计模式中最为简单的一种模式,甚至有些模式大师都不称其为模式,称其为一种实现技巧,因为设计模式讲究对象之间的关系的抽象,而单例模式只有自己一个对象,也因此有些设计大师并把把其称为设计模式之一. 这里又不具体讲如何实现单例模式和介绍其原理(因为这方便的已经有太多的好文章介绍了),如果对单例模式不了解的可以先看下:http://terrylee.cnblogs.com/archive/2005/12/09/293509.html . 好多没怎么使用过的人

(转)设计模式之——单例模式(Singleton)的常见应用场景

单例模式(Singleton)也叫单态模式,是设计模式中最为简单的一种模式,甚至有些模式大师都不称其为模式,称其为一种实现技巧,因为设计模式讲究对象之间的关系的抽象,而单例模式只有自己一个对象,也因此有些设计大师并把把其称为设计模式之一. 这里又不具体讲如何实现单例模式和介绍其原理(因为这方便的已经有太多的好文章介绍了),如果对单例模式不了解的可以先看下:http://terrylee.cnblogs.com/archive/2005/12/09/293509.html .当然也可以自己搜索.

设计模式之单例模式Singleton pattern

单例模式Singleton pattern 一种软件设计模式.在核心结构中只包含一个被称为单例的特殊类. 一个类只有一个对象实例,并且自行实例化向整个系统提供. 动机 一个系统中可以存在多个打印任务,但是只能有一个正在工作的任务:一个系统只能有一个窗口管理器或文件系统:一个系统只能有一个计时工具或ID(序号)生成器.如在Windows中就只能打开一个任务管理器.如果不使用机制对窗口对象进行唯一化,将弹出多个窗口,如果这些窗口显示的内容完全一致,则是重复对象,浪费内存资源:如果这些窗口显示的内容不

一天一个设计模式(二) -单例模式(Singleton)

前言 单例模式 (Singleton) 是一种创建型模式,指某个类采用Singleton模式,则在这个类被创建后,只可能产生一个实例供外部访问,并且提供一个全局的访问点. 正文 (一). 优缺点 Java中单例模式 (Singleton) 是一种广泛使用的设计模式.单例模式的主要作用是保证在Java程序中,某个类只有一个实例存在.一些管理器和控制器常被设计成单例模式. 1. 优点 提供了对唯一实例的受控访问. 由于在系统内存中只存在一个对象,因此可以节约系统资源,对于一些需要频繁创建和销毁的对象

二十四种设计模式:单例模式(Singleton Pattern)

单例模式(Singleton Pattern) 介绍保证一个类仅有一个实例,并提供一个访问它的全局访问点. 示例保证一个类仅有一个实例. Singleton using System; using System.Collections.Generic; using System.Text; namespace Pattern.Singleton { /// <summary> /// 泛型实现单例模式 /// </summary> /// <typeparam name=&q

Android设计模式之单例模式(Singleton Pattern)

个人总结学习和研究,部分内容参考<Android源码设计模式解析与实战>一书~~ 一.  定义: 确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例. 也就是说,单例要满足3点: 1.单例类只能有一个实例. 2.单例类必须自己创建自己的唯一实例.(构造函数私有化,防止外部程序通过new来构造). 3.单例类必须给其他对象提供这一实例.(暴露公有静态方法或者通过枚举返回单例类对象). 二.  使用场景: 确保某个类有且只有一个对象的场景,避免产生多个对象消耗过多的资源.比如说在一个

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

一 单例模式介绍及它的使用场景 单例模式是应用最广的模式,也是我最先知道的一种设计模式.在深入了解单例模式之前.每当遇到如:getInstance()这样的创建实例的代码时,我都会把它当做一种单例模式的实现. 事实上常常使用的图片载入框架ImageLoader的实例创建就是使用了单例模式.由于这个ImageLoader中含有线程池.缓存系统.网络请求,非常消耗资源,不应该创建多个对象,这时候就须要用到单例模式. ImageLoader的创建代码例如以下: ImageLoader.getInsta