设计模式-单例模式

简介: 设计模式-单例模式

单例模式


有如下五中方式可以实现单例模式


饿汉

概念

恶汉模式,在成员变量声明时就会创建对象,也就是类被首次加载时就会创建对象


优劣势

优:

  • 不存在线程安全问题

劣:

  • 因为首次加载即创建,所以如果对象较大可能会拖慢程序启动速度


代码

public class HungrySingletom implements Print {
    private HungrySingletom() {
    }
    private static HungrySingletom INSTANCE = new HungrySingletom();
    public static HungrySingletom getInstance() {
        return INSTANCE;
    }
    @Override
    public void printSomeThing() {
        System.out.println("恶汉打印数据");
    }
}
class Client {
    public static void main(String[] args) {
        HungrySingletom.getInstance().printSomeThing();
    }
}
复制代码


懒汉

需要的时候才去创建,缺点是默认线程不安全


线程不安全懒汉

概念

初始变量为空,当我们获取的时候如果静态实例为空则创建一个对象。

如果我们在多线程中去获取单例对象则是线程不安全的,极有可能会创建多个实例

代码

本例中我们给getInstance方法加上了synchronized关键字,这样就可以保证了获取单例方法的线程安全。

在调用的时候我们会for循环创建200个线程打印对象地址,最终结果打印对象地址相同,没有出现线程不安全问题。

public class LazyThreadSafeSingleTon implements Print, Print1 {
    private static LazyThreadSafeSingleTon instance = null;
    private LazyThreadSafeSingleTon() {
    }
    public static synchronized LazyThreadSafeSingleTon getInstance() {
        if (instance == null) {
            instance = new LazyThreadSafeSingleTon();
        }
        return instance;
    }
    @Override
    public void printSomeThing() {
        System.out.println("线程安全懒汉单例");
    }
    public static void main(String[] args) {
        for (int i = 0; i < 200; i++) {
            final int index = i;
            new Thread(new Runnable() {
                @Override
                public void run() {
                    LazyThreadSafeSingleTon myInstance = LazyThreadSafeSingleTon.getInstance();
                    myInstance.printMsg("索引:" + index + "   " + myInstance.toString());
                }
            }).start();
        }
    }
    @Override
    public void printMsg(String msg) {
        System.out.println(msg);
    }
}
复制代码


线程安全懒汉

方法加锁实现线程安全单例

在获取单例的方法上加上synchronized锁,实现线程安全

优劣势

优势: 使用的时候再初始化,可以加快初始化时间

劣势 在方法上加上了synchronized所以每次获取实例的时候都会加锁,当频繁获取实例的时候会明显降低效率。解决办法是使用双检索单例。

代码
public class LazyThreadSafeSingleTon implements Print, Print1 {
    private static LazyThreadSafeSingleTon instance = null;
    private LazyThreadSafeSingleTon() {
    }
    public static synchronized LazyThreadSafeSingleTon getInstance() {
        if (instance == null) {
            instance = new LazyThreadSafeSingleTon();
        }
        return instance;
    }
    @Override
    public void printSomeThing() {
        System.out.println("线程安全懒汉单例");
    }
    public static void main(String[] args) {
        for (int i = 0; i < 200; i++) {
            final int index = i;
            new Thread(new Runnable() {
                @Override
                public void run() {
                    LazyThreadSafeSingleTon myInstance = LazyThreadSafeSingleTon.getInstance();
                    myInstance.printMsg("索引:" + index + "   " + myInstance.toString());
                }
            }).start();
        }
    }
    @Override
    public void printMsg(String msg) {
        System.out.println(msg);
    }
}
复制代码


双检索方式实现单例

双检索代码理解

一下文字是对下面代码的说明

线程1第一次获取实例 线程第一次调用的时候会先进入第一重判断,此时instance为空,进入加锁代码。 然后进入第二重判断,此时instance仍然为空,开始初始化。

线程2在线程1初始化过程中获取实例 因为实例尚没有初始化,所以线程2第一重判断instance为空,二此时线程1正在初始化所以线程2要等待锁。延迟后线程1释放锁,线程1进入锁内并进行第二重判断。因为线程1已经初始化了,所以此时instance不为空,所以不进入第二重判断。直接跳到返回实例的代码

线程3在线程1、2完成后进入 线程3进入第一重判断,此时instance不为空,直接跳到返回instance部分

为什么一定要加volatie关键字

将我很久以前写的列举发一下吧,目前我仍然觉得写的没问题


代码
public class DoubleLockThreadSafeSingleTon implements Print1 {
    private static volatile DoubleLockThreadSafeSingleTon instance;//volatie是必须的
    private DoubleLockThreadSafeSingleTon(){}
    public static DoubleLockThreadSafeSingleTon getInstance() {
        if(instance==null){//第一重判断
            synchronized (DoubleLockThreadSafeSingleTon.class){//加锁
                if (instance==null){//第二重判断
                    instance=new DoubleLockThreadSafeSingleTon();
                }
            }
        }
        return instance;
    }
    public static void main(String[] args) {
        for (int i=0;i<200;i++){
            System.out.println(DoubleLockThreadSafeSingleTon.getInstance().toString());
        }
    }
    @Override
    public void printMsg(String msg) {
        System.out.println(msg);
    }
}
复制代码


静态内部类

理解

本例代码中StaticClassSingleton实例在StaticClassSingletonHolder的静态变量中初始化


优劣势

优势 在不使用锁的情况下实现了单例模式

劣势 需要多创建一个内部类


代码

public class StaticClassSingleton implements Print {
    private StaticClassSingleton() {
    }
    public static StaticClassSingleton getInstance() {
        return StaticClassSingletonHolder.instance;
    }
    static class StaticClassSingletonHolder {
        static StaticClassSingleton instance = new StaticClassSingleton();
    }
    @Override
    public void printSomeThing() {
        System.out.println("静态内部类实现单例");
    }
    public static void main(String[] args) {
        StaticClassSingleton.getInstance().printSomeThing();
    }
}
复制代码


枚举

很多人可能是不知道的,枚举实例中是可以编写方法代码的,根据这种方式我们可以巧妙的实现单例模式。


优劣势

优势 不需要锁就可以实现单例劣势 暂未想到


代码

public enum EnumSingleton implements Print {
    INSTANCE,DEFAULT;
    @Override
    public void printSomeThing() {
        System.out.println(name());
    }
    public static void main(String[] args) {
        EnumSingleton.INSTANCE.printSomeThing();
        EnumSingleton.DEFAULT.printSomeThing();
    }
}
复制代码


扩展   枚举代码变种实现路由

我们可以在枚举中定义方法,然后再实例中重写该方法,从而实现不同的枚举实例有自己的行为

public enum EnumSingleton implements Print {
    INSTANCE{
        @Override
        public void action(){
            System.out.println("INSTANCE单例方法");
        }
    },DEFAULT{
        @Override
        public void action(){
            System.out.println("DEFAULT单例方法");
        }
    };
    @Override
    public void printSomeThing() {
        System.out.println(name());
    }
    public static void main(String[] args) {
        EnumSingleton.INSTANCE.printSomeThing();
        EnumSingleton.DEFAULT.printSomeThing();
        EnumSingleton.INSTANCE.action();
        EnumSingleton.DEFAULT.action();
    }
    public void action() {
    }
}



相关文章
|
设计模式 缓存 安全
【设计模式】【创建型模式】单例模式(Singleton)
一、入门 什么是单例模式? 单例模式是一种设计模式,确保一个类只有一个实例,并提供一个全局访问点。它常用于需要全局唯一对象的场景,如配置管理、连接池等。 为什么要单例模式? 节省资源 场景:某些对象创
448 15
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
165 2
|
设计模式 安全 Java
设计模式:单例模式
单例模式是一种创建型设计模式,确保一个类只有一个实例,并提供全局访问点。它通过私有化构造函数、自行创建实例和静态方法(如`getInstance()`)实现。适用于数据库连接池、日志管理器等需要全局唯一对象的场景。常见的实现方式包括饿汉式、懒汉式、双重检查锁、静态内部类和枚举。线程安全问题可通过`synchronized`或双重检查锁解决,同时需防止反射和序列化破坏单例。优点是避免资源浪费,缺点是可能增加代码耦合度和测试难度。实际开发中应优先选择枚举或静态内部类,避免滥用单例,并结合依赖注入框架优化使用。
|
设计模式 存储 安全
设计模式2:单例模式
单例模式是一种创建型模式,确保一个类只有一个实例,并提供全局访问点。分为懒汉式和饿汉式: - **懒汉式**:延迟加载,首次调用时创建实例,线程安全通过双重检查锁(double check locking)实现,使用`volatile`防止指令重排序。 - **饿汉式**:类加载时即创建实例,线程安全但可能浪费内存。 示例代码展示了如何使用Java实现这两种模式。
329 4
|
设计模式 存储 安全
设计模式-单例模式练习
单例模式是Java设计模式中的重要概念,确保一个类只有一个实例并提供全局访问点。本文详解单例模式的核心思想、实现方式及线程安全问题,包括基础实现(双重检查锁)、懒汉式与饿汉式对比,以及枚举实现的优势。通过代码示例和类图,深入探讨不同场景下的单例应用,如线程安全、防止反射攻击和序列化破坏等,展示枚举实现的简洁与可靠性。
226 0
|
设计模式 存储 前端开发
前端必须掌握的设计模式——单例模式
单例模式是一种简单的创建型设计模式,确保一个类只有一个实例,并提供一个全局访问点。适用于窗口对象、登录弹窗等场景,优点包括易于维护、访问和低消耗,但也有安全隐患、可能形成巨石对象及扩展性差等缺点。文中展示了JavaScript和TypeScript的实现方法。
672 13
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
236 2
|
设计模式 Java 数据库连接
Java编程中的设计模式:单例模式的深度剖析
【10月更文挑战第41天】本文深入探讨了Java中广泛使用的单例设计模式,旨在通过简明扼要的语言和实际示例,帮助读者理解其核心原理和应用。文章将介绍单例模式的重要性、实现方式以及在实际应用中如何优雅地处理多线程问题。
260 4
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
|
设计模式 存储 数据库连接
PHP中的设计模式:单例模式的深入理解与应用
【10月更文挑战第22天】 在软件开发中,设计模式是解决特定问题的通用解决方案。本文将通过通俗易懂的语言和实例,深入探讨PHP中单例模式的概念、实现方法及其在实际开发中的应用,帮助读者更好地理解和运用这一重要的设计模式。
209 1