设计模式之单例模式

单例模式

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

特征

  • 构造方法一般不外开放

  • 通过一个静态方法或枚举返回单例类的对象

  • 注意多线程的场景

  • 注意单例类对象在反序列化时不会重新创建对象


单例类的实现

1.懒汉式单例

    // 1.懒汉式单例
    static class Singleton1 {

        // 私有静态变量
        private static Singleton1 mInstance = null;

        // 私有构造方法
        private Singleton1() {
            System.out.println("懒汉式单例");
        }

        // 暴露公有静态方法
        public static Singleton1 getInstance() {
            if (mInstance == null) {
                mInstance = new Singleton1();
            }
            return mInstance;
        }
    }

以上是简单的懒汉式单例实现,如果单例类中有作网络请求、缓存操作或线程池策略等耗时操作,会创建多个不同的实例,以下稍作改动,模拟耗时操作。

main()中添加

        for (int i = 0; i < 100; i++) {
            final int index = i;
            new Thread(){
                @Override
                public void run() {
                    System.out.println("index = " + index + ", instance: " + Singleton1.getInstance());
                }
            }.start();
        }

getInstance()中添加,再运行结果

try {
     Thread.sleep(500);
} catch (InterruptedException e) {
    e.printStackTrace();
}

---------- 节选结果 ----------
index = 1, instance: com.example.designmodedemo.singleton.SingletonMode$Singleton1@6b52350c
懒汉式单例
懒汉式单例
懒汉式单例
index = 7, instance: com.example.designmodedemo.singleton.SingletonMode$Singleton1@4a1a7268
懒汉式单例
懒汉式单例
index = 9, instance: com.example.designmodedemo.singleton.SingletonMode$Singleton1@ce407e7
懒汉式单例
懒汉式单例
懒汉式单例
index = 11, instance: com.example.designmodedemo.singleton.SingletonMode$Singleton1@59ac37b1
index = 8, instance: com.example.designmodedemo.singleton.SingletonMode$Singleton1@1dca18a4
index = 13, instance: com.example.designmodedemo.singleton.SingletonMode$Singleton1@5795dfe4
index = 10, instance: com.example.designmodedemo.singleton.SingletonMode$Singleton1@7898c39
index = 6, instance: com.example.designmodedemo.singleton.SingletonMode$Singleton1@fb49a4f
index = 5, instance: com.example.designmodedemo.singleton.SingletonMode$Singleton1@47c48106
index = 3, instance: com.example.designmodedemo.singleton.SingletonMode$Singleton1@6fada00f

输出的哈希地址不相等说明对象多次创建。想要线程安全,需要加synchronized关键字。

1、直接锁方法。锁方法的缺陷是同步锁粒度太大,耗性能

2、双重校验DCL。同步锁粒度小很多。由于JVM有指令重排特殊性,会导致双重校验失效,静态变量需要添加volatile关键字

private volatile static Singleton1 sInstance = null;

public static Singleton1 getInstance() {
    if (sInstance == null) {// 第一重校验
        try {
            Thread.sleep(500);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        synchronized(Singleton1.class) {
            if (sInstance == null) {// 第二重校验
                sInstance = new Singleton1();
            }
        }
    }
    return sInstance;
}

2.饿汉式单例(线程安全)

    static class Singleton2 {

        private static Singleton2 sInstance = new Singleton2();

        private Singleton2() {}

        public static Singleton2 getInstance() {
            try {
                Thread.sleep(500);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            return sInstance;
        }
    }

3.静态内部类单例(线程安全)

static class Singleton3 {

    private Singleton3() {}

    private static class Holder {
        private static final Singleton3 INSTANCE = new Singleton3();
    }

    // 延时加载
    public final static Singleton3 getInstance() {
        return Holder.INSTANCE;
    }
}

4.使用枚举(线程安全)

static class Singleton4 {}

enum EnumDemo {
    INSTANCE;

    private Singleton4 instance;

    EnumDemo() {
        instance = new Singleton4();
    }

    public Singleton4 getInstance() {
        return instance;
    }
}

4.使用容器,如源码中类SystemServiceRegistry

static class Singleton5 {}

static class SingletonManager {
    private static Map<String, Object> map = new HashMap<>();

    public static void putClass(String key, Object instace) {
        if (!map.containsKey(key)) {
            map.put(key, instace);
        }
    }

    public static Object getClass(String key) {
        return map.get(key);
    }
}

学习笔记,如有错漏,理解不到位,烦请多多提点,拜谢!

若需转载,声明出处即可

简书 && github


展开阅读全文

图解Java设计模式(一)

02-27
1) 优秀的程序应该是这样的:阅读时,感觉很优雅;新增功能时,感觉很轻松;运行时,感觉很快速,这就需要设计模式支撑。 2) 设计模式包含了大量的编程思想,讲授和真正掌握并不容易,网上的设计模式课程不少,大多讲解的比较晦涩,没有真实的应用场景和框架源码支撑,学习后,只知其形,不知其神。就会造成这样结果: 知道各种设计模式,但是不知道怎么使用到真实项目。本课程针对上述问题,有针对性的进行了升级 (1) 授课方式采用 图解+框架源码分析的方式,让课程生动有趣好理解 (2) 系统全面的讲解了设计模式,包括 设计模式七大原则、UML类图-类的六大关系、23种设计模式及其分类,比如 模式的8种实现方式、工厂模式的3种实现方式、适配器模式的3种实现、代理模式的3种方式、深拷贝等 3) 如果你想写出规范、漂亮的程序,就花时间来学习下设计模式吧 课程内容和目标 本课程是使用Java来讲解设计模式,考虑到设计模式比较抽象,授课采用 图解+框架源码分析的方式 1) 内容包括: 设计模式七大原则(一职责、接口隔离、依赖倒转、里氏替换、开闭原则、迪米特法则、合成复用)、UML类图(类的依赖、泛化和实现、类的关联、聚合和组合) 23种设计模式包括:创建型模式模式(8种实现)、抽象工厂模式、原型模式、建造者模式、工厂模式。结构型模式:适配器模式(3种实现)、桥接模式、装饰模式、组合模式、外观模式、享元模式、代理模式(3种实现)。行为型模式:模版方法模式、命令模式、访问者模式、迭代器模式、观察者模式、中介者模式、备忘录模式、解释器模式(Interpreter模式)、状态模式、策略模式、职责链模式(责任链模式) 2) 学习目标:通过学习,学员能掌握主流设计模式,规范编程风格,提高优化程序结构和效率的能力。

全新150集通俗易懂的Java设计模式

07-01
本教程为授权出品教程 1) 优秀的程序应该是这样的:阅读时,感觉很优雅;新增功能时,感觉很轻松;运行时,感觉很快速,这就需要设计模式支撑 2) 设计模式包含了大量的编程思想,讲授和真正掌握并不容易,网上的设计模式课程不少,大多讲解的比较晦涩,没有真实的应用场景和框架源码支撑,学习后,只知其形,不知其神。就会造成这样结果: 知道各种设计模式,但是不知道怎么使用到真实项目。本课程针对上述问题,有针对性的进行了升级 (1) 授课方式采用 图解+框架源码分析的方式,让课程生动有趣好理解 (2) 系统全面的讲解了设计模式,包括 设计模式七大原则、UML类图-类的六大关系、23种设计模式及其分类,比如 模式的8种实现方式、工厂模式的3种实现方式、适配器模式的3种实现、代理模式的3种方式、深拷贝等 3) 如果你想写出规范、漂亮的程序,就花时间来学习下设计模式吧 课程内容和目标 本课程是使用Java来讲解设计模式,考虑到设计模式比较抽象,授课采用 图解+框架源码分析的方式 1) 内容包括:设计模式七大原则(一职责、接口隔离、依赖倒转、里氏替换、开闭原则、迪米特法则、合成复用)、UML类图(类的依赖、泛化和实现、类的关联、聚合和组合) 23种设计模式包括:创建型模式模式(8种实现)、抽象工厂模式、原型模式、建造者模式、工厂模式。结构型模式:适配器模式(3种实现)、桥接模式、装饰模式、组合模式、外观模式、享元模式、代理模式(3种实现)。行为型模式:模版方法模式、命令模式、访问者模式、迭代器模式、观察者模式、中介者模式、备忘录模式、解释器模式(Interpreter模式)、状态模式、策略模式、职责链模式(责任链模式)。 2) 学习目标:通过学习,学员能掌握主流设计模式,规范编程风格,提高优化程序结构和效率的能力。
©️2020 CSDN 皮肤主题: 大白 设计师: CSDN官方博客 返回首页
实付0元
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值