设计模式 - 对象性能之单例模式

一、对象性能模式

面向对象很好地解决了”抽象“的问题,但是必不可免地要付出一定的代价。对于通常情况来讲,面向对象的成本大都可以忽略不计。但是某些情况,面向对象所带来的成本必须谨慎处理。

典型模式:Singleton、Flyweight

二、动机

在软件系统中,经常有这样一些特殊的类,必须保证它们再系统中只存在一个实例,才能确保它们的逻辑正确性,以及良好的效率

如何绕过常规的构造器,提供一种机制来保证一个类只有一个实例?

这应该是设计者的责任,而不是使用者的责任

三、模式定义

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

四、结构

 

五、要点总结

Singleton 模式中的实例构造器可以设置为 protected 以允许子类派生

Singleton 模式一般不要支持拷贝构造函数和 Clone 接口,因为这有可能导致多个对象实例

如何实现多线程环境下的安全的Singleton?注意对双检查锁的正确实现

六、代码示例

6.1 懒汉式(非线程安全)

public class Singleton implements Serializable {
    private Singleton3() {
        // ... 构造方法
    }

    private static Singleton INSTANCE = null;

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

}

6.2 线程安全(性能低)

public class Singleton implements Serializable {
    private Singleton3() {
        // ... 构造方法
    }

    private static Singleton INSTANCE = null;

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

}

6.3 双检锁懒汉式

public class Singleton implements Serializable {
    private Singleton() {
        
    }

    private static volatile Singleton INSTANCE = null; // 可见性,有序性

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

}

加锁前判断是否为 null,是为了提高性能。

加锁后判断是否为 null,是为了防止两个线程同时进入 if (INSTANCE == null) 代码块,若不再判断一次是否为 null,会执行两次构造函数

加 volatile 关键字是为了防止构造函数的指令重排序,若不加则可能先将分配的空间地址赋值给INSTANCE ,再进行构造器。此时另一个线程判断 INSTANCE 不为空,会返回 INSTANCE,但INSTANCE 状态并未构造完毕。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值