设计模式之单例模式

单例模式(Singleton Pattern)是最简单的一种设计模式。下面让我们开始学习单例模式。

一、基本介绍

单例模式模式确保一个类只有一个实例,并且自行实例化并向整个系统提供该实例。
单例模式的主要作用是确保一个类只有一个实例的存在。单例模式可以用在建立目录,数据库连接等需要单线程操作的场合,用于实现对系统资源的控制。

java中的单例模式一般有两种表现形式:

  1. 饿汉模式: 类加载时,就进行对象实例化
  2. 懒汉模式: 第一次引用类时,才进行对象实例化

二、饿汉模式

下面介绍饿汉模式的创建步骤

1 . 创建私有静态成员变量并实例化

2 . 将构造函数私有化,使外界无法直接实例化

3 . 提供公开方法返回创建的实例化对象

下面是创建饿汉模式的代码

public class Singleton {
  //1.创建私有静态成员变量并实例化
  private static Singleton mSingleton = new Singleton();
  //2.将构造函数私有化,使外界无法直接实例化
  private Singleton(){}
  //3.提供公开方法返回创建的实例化对象
  public static Singleton getInstance() {
  return mSingleton;
}

使用的时候代码如下:

//获得唯一的实例
Singleton singleton = Singleton.getInstance();

注意:由于我们的构造方法是私有化的,所以外界不能调用它,所以单例模式的类是不能被继承的。

三、懒汉模式

下面介绍懒汉模式的创建步骤

1 . 声明私有的静态成员变量

2 . 将构造函数实例化,使外界无法直接实例化

3 . 提供公开的方法,判断如果实例为null则创建实例并返回,不为空则直接返回实例

4 . 给方法添加synchronized关键字,保证其在多线程环境下也只创建一个实例

下面是创建懒汉模式的额代码

public class Singleton2 {
    //1.声明私有的静态成员变量
    private static Singleton2 mSingleton2 = null;
    //2.将构造函数实例化,使外界无法直接实例化
    private Singleton2() {}
    //3.提供公开的方法,判断如果实例为null则创建实例并返回,不为空则直接返回实例
    //4.给方法添加synchronized关键字,保证其在多线程环境下也只创建一个实例
    synchronized public static Singleton2 getInstance() {
        if(mSingleton2 == null){
            mSingleton2 = new Singleton2();
        }
        return mSingleton2;
    }
}

使用的时候代码如下

Singleton2 singleton2 = Singleton2.getInstance();

注意:懒汉模式中需要使用synchronized关键字对静态方法进行同步,保证其在多线程环境中只创建一个实例。加入我们没有对该方法进行同步,如果线程A和线程B同时调用该方法,那么它们都会判断得到mSingleton2为空,那么线程A和B就会各自创建一个对象,那么在线程中就会存在两个对象,这就违背了单例模式的定义,所以必须使用synchronized关键字对线程进行同步。

四、饿汉模式和懒汉模式的对比

下面通过一张表格来对比两种实现方式

模式类型 实例化时间 获取对象的速度 实现难度 线程安全
饿汉模式 类被加载时 较快 在C++内不容易实现,java容易实现 线程安全
懒汉模式 第一次引用时 较慢 容易实现 线程不安全,需要synchronized关键字同步方法

五、单例模式的优缺点

1 . 单例模式的优点:

  • 由于单例模式的内存中只有一个实例,减少了内存的开支,特别是一个对象需要频繁地创建、销毁,并且创建和销毁的性能又无法优化时,单例模式就非常有优势

  • 由于单例模式只生成一个实例,所以减少了系统的性能开销,当一个对象的产生需要比较多的资源时,如读取配置,产生其他依赖对象时,可以通过在启用时直接产生一个单例对象,然后永久驻留内存的方式来解决

  • 单例模式可以避免对资源的多重占用。例如对于一个写文件的操作,由于只有一个实例存在于内存中,避免了对同一个资源文件的同时操作

  • 单例模式可以在系统设置全局的访问点,优化和共享资源访问,比如可以设计一个单例类负责所有数据表的映射处理

2 . 单例模式的缺点:

  • 单例模式无法创建子类,扩展困难,要想扩展就得修改代码

  • 单例模式对测试不利,在并行开发中,如果单例模式的类没有完成,就没法进行测试

  • 单例模式与单一职责原则有冲突。单例模式把要单例业务逻辑融合在了一起

六、单例模式的典型使用场景

1 . 要求生成唯一序列号的环境

2 . 在一个项目中需要共享访问点或共享数据。比如一个Web页面上的访问次数计数器可以不用在每次页面被别人访问时就去更新数据库的值,可以用一个单例模式来记录页面的访问次数,并确保单例模式的线程安全

3 . 创建一个对象需要消耗很多资源。比如访问IO或数据库

4 . 需要定义大量的静态常量和静态方法的环境,可以使用单例模式(当然也可以直接将常量和方法声明为静态的)

七、单例模式的使用注意事项

根据功能单例类又可以分为有状态的单例类和无状态的单例类

1 . 有状态的单例类的单例类的对象是可变的,通常被当做状态库使用,比如给系统提供一个唯一的序列号,每次提供的序列号是变化的,但序列号是全局唯一的。

2 . 无状态的单例类是不变的,通常用来提供工具性的功能方法,比如IO或数据库访问

单例模式使用时必须注意以下三点:
1 . 单例类仅仅局限于一个JVM,因此当处于有多个JVM的分布式系统时,这个单例类就会在多个JVM中被实例化,就会出现多个对象。如果是无状态的单例类,比如数据库访问,是没有问题的,而如果是有状态的单例类,比如一个用来产生唯一序列号的单例类,这种多个JVM的情况就会造成产生的序列号可能不唯一,因此在任何使用EJB、RMI和JINI技术的分布式系统中应该避免使用有状态的单例类。

2 . 当一个JVM有多个类加载器时,多个类加载器会同时加载同一个实例,此时就会产生多个实例,此时应该避免使用有状态的单例类

3 . 注意序列化和克隆对实例唯一性的影响,如果一个单例类实现了Serializable或Cloneable接口,则有可能被反序列化或克隆出一个实例来,也会破坏单例模式的定义

八、单例模式使用举例

上面说了这么多理论性的东西,下面我们将通过一个单例模式记录访问次数的例子来展示单例模式的应用。

先创建一个单例类

/**
 * 使用饿汉模式创建一个单例类
 * 使用synchronized对方法进行线程同步
 */
public class GlobalNum {
    private static GlobalNum globalNum = new GlobalNum();
    private int num = 0;
    public static GlobalNum getInstance() {
        return globalNum;
    }

    public synchronized int getNum() {
        return ++num;
    }
}

然后我们使用两个线程去模拟访问次数的增加

public class SingletonTest {
    public static void main(String[] args) {
        //创建线程A
        NumThread numThreadA = new NumThread("线程A");
        //创建线程B
        NumThread numThreadB = new NumThread("线程B");
        //启动线程
        numThreadA.start();
        numThreadB.start();
    }
}

class NumThread extends Thread{
    private String threadName;

    public NumThread(String threadName) {
        this.threadName = threadName;
    }

    @Override
    public void run() {
        GlobalNum globalNum = GlobalNum.getInstance();
        for (int i = 0; i < 5; i++) {
            System.out.println(threadName + "第" + globalNum.getNum() + "次访问");

            try {
                //线程休眠1秒
                this.sleep(1000);
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }
    }
}

运行结果如下:

线程A第1次访问
线程B第2次访问
线程A第3次访问
线程B第4次访问
线程B第6次访问
线程A第5次访问
线程B第7次访问
线程A第8次访问
线程A第9次访问
线程B第10次访问

可以看出它保证了两个线程使用的对象是同一个对象,从而来记录访问的次数

九、后记

以上就是关于单例模式的介绍,希望可以帮助到需要的人。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容