《Java-SE-第二十三章》之单例模式

简介: 《Java-SE-第二十三章》之单例模式

文章目录

单例模式概述

饿汉模式

懒汉模式单线程版

懒汉单例多线程版

枚举实现单例

单例模式概述

单例模式是设计模式中的一种,其作用能保证某个类在程序中只存在唯一一份实例,而不会创建多份实例。单例模式具体的实现方式, 分成 “饿汉” 和 “懒汉” 两种.。饿汉模式中的饿不并不是真饿了,而是说提前把单例类更创建好。懒汉模式中的懒则是当需要使用到单例类的时候才创建单例对象。这就类似于,每次吃饭的时候,已经提前把碗洗了有碗用,这就是"饿汉"。"懒汉"则是要用碗赶紧去把碗洗了。单例模式中的单例类有且只有一份,static修饰的成员变量是属于类的,也是只有一份,所以我们使用static修饰的成员变量保存 到实例对象的引用变量。

饿汉模式

在单例模式中一个类只有一个实例对象,所以避免外部通过构造方法创建对象,所以需要才构造方法私有,防止创建出多个对象。又因为我们是使用static来保存实例对象的引用,所以需要提供一个静态方法获取唯一的对象。

示例代码

public class HungrySingleton {
    /**
     * 类加载的时候创建出HungrySingleton对象,并用静态差成员变量保存。
     */
    private static HungrySingleton singleton = new HungrySingleton();
    private HungrySingleton() {
    }
    /**
     * 提供静态方法获取单例
     * @return
     */
    public static HungrySingleton getSingleton() {
        return singleton;
    }
}

上述的代码多线程环境下依旧是安全的,因为 getSingleton()方法中的return语句只涉及到读而不涉及修改。

懒汉模式单线程版

懒汉模式和饿汉模式相比就是创建实例的时机不一样,懒汉不是在类加载的时候创建而是在需要使用的时候创建。

示例代码

public class SlackerSingleton {
    private static SlackerSingleton singleton = null;
    private SlackerSingleton() {
    }
    /**
     * 提供静态方法获取单例
     *
     * @return
     */
    public static SlackerSingleton getSingleton() {
        //需要使用的时候创建对象
        if (singleton == null) {
            singleton = new SlackerSingleton();//真正创建的时机,创建好了,后续是直接返回。
        }
        return singleton;
    }
}

多线程的环境下,当t1线程进行比较singleton是否为空时,比较完之后,t1线程还没有创建实例。此时t2线程立马进来再次判断此时singleton依旧为空,导致t2线程也进行new操作,最终导致创建了多份实例。造成线程不安全的代码就是if语句以及实例的创建操作,所以我们需要对这段代码加锁。

懒汉单例多线程版

加锁后的代码

public class SlackerSingleton {
    private static SlackerSingleton singleton = null;
    private SlackerSingleton() {
    }
    /**
     * 提供静态方法获取单例
     *
     * @return
     */
    public static SlackerSingleton getSingleton() {
        //需要使用的时候创建对象
        synchronized (SlackerSingleton.class) {
            if (singleton == null) {
                singleton = new SlackerSingleton();
            }
        }
        return singleton;
    }
}

上述代码仅仅是针对首次创建实例的情况,如果singleton已经创建好了,if语句就啥用了,但是第二次进入该方法会去获取锁,而获取锁海外释放锁的是耗时耗力的。所以可以在加锁之前进行判断该实例是否已经创建好了,没有则获取锁,创建好了直接返回。

双重if的代码

public class SlackerSingleton {
    private static SlackerSingleton singleton = null;
    private SlackerSingleton() {
    }
    /**
     * 提供静态方法获取单例
     *
     * @return
     */
    public static SlackerSingleton getSingleton() {
        //需要使用的时候创建对象
        if (singleton == null) {//判断是否需要加锁
            synchronized (SlackerSingleton.class) {
                if (singleton == null) {//判断是否需要创建实例
                    singleton = new SlackerSingleton();
                }
            }
        }
        return singleton;
    }
}

优化到这里依旧是存在问题的,和这个锅就是编译器的了。上述代码需要判断isingleton == null,这个操作在多线程的情况下是非常频繁的,导致CPU频繁的从内存读取(Load)到寄存器然后进行比较(CMP),此时就会进行优化,线程不会去读取内存中数据,而是会从寄存器中读取数据,一旦当singleton值变化时,线程是感知不到的,就会造成内存可见性问题,除此之外,还有另一个问题就是由new SlackerSingleton();引起来的,创建对象大概可以分为三步:①:JVM为对象分配一块内存S,②:在内存S上对对象进行初始化,③:将内存S的地址赋值singleton变量。为了引出bug,我们假设有两个线程t1和t2同时使用getSingleton()方法。

正常情况按照对象创建①②③来执行。

第一步:线程t1直接运行到singleton = new SlackerSingleton();并且一口气执行完了①②③。

第二步:线程t2进行该方法判断singleton就直接返回了。

第三步:线程t2可以正常使用。

出现bug的情况按对象创建的步骤①③②来执行。

第一步:线程t1执行singleton = new SlackerSingleton();执行完①JVM为对象分配一块内存.③将内存的地址赋值给singleton 变量

第二步:线程t2进入该方法进行判断发现singleton不为空,拿到singleton 返回了。此时线程t2拿到的是singleton 的半成品对象,因为该对象没有初始化。

第三步:线程t2拿到该对象去使用就可能会出现异常。

综上所述,还存在两个隐含问题一个是内存可见性问题,另一个是指令重排序问题。解决这个问题只需要将静态变量加上volatile 即可。

volatile 修饰静态变量

public class SlackerSingleton {
    private static volatile SlackerSingleton singleton = null;
    private SlackerSingleton() {
    }
    /**
     * 提供静态方法获取单例
     *
     * @return
     */
    public static SlackerSingleton getSingleton() {
        //需要使用的时候创建对象
        if (singleton == null) {
            synchronized (SlackerSingleton.class) {
                if (singleton == null) {
                    singleton = new SlackerSingleton();
                }
            }
        }
        return singleton;
    }
}

此时多线程长场景下的懒汉模式完成。

枚举实现单例

因为枚举是天然的单例,并且枚举类型无法通过反射来获取封装的私有变量,非常安全。

示例代码

public enum Singleton {
     INSTANCE;
     public void businessMethod() {
          System.out.println("我是一个单例!");
     }
}

简单使用

public class SingletonDemo {
    public static void main(String[] args) {
        Singleton.INSTANCE.businessMethod();
    }
}

运行结果

相关文章
|
3月前
|
设计模式 缓存 安全
Java设计模式的单例模式应用场景
Java设计模式的单例模式应用场景
41 4
|
24天前
|
设计模式 存储 负载均衡
【五】设计模式~~~创建型模式~~~单例模式(Java)
文章详细介绍了单例模式(Singleton Pattern),这是一种确保一个类只有一个实例,并提供全局访问点的设计模式。文中通过Windows任务管理器的例子阐述了单例模式的动机,解释了如何通过私有构造函数、静态私有成员变量和公有静态方法实现单例模式。接着,通过负载均衡器的案例展示了单例模式的应用,并讨论了单例模式的优点、缺点以及适用场景。最后,文章还探讨了饿汉式和懒汉式单例的实现方式及其比较。
【五】设计模式~~~创建型模式~~~单例模式(Java)
|
13天前
|
设计模式 安全 Java
Java 单例模式,背后有着何种不为人知的秘密?开启探索之旅,寻找答案!
【8月更文挑战第30天】单例模式确保一个类只有一个实例并提供全局访问点,适用于需全局共享的宝贵资源如数据库连接池、日志记录器等。Java中有多种单例模式实现,包括饿汉式、懒汉式、同步方法和双重检查锁定。饿汉式在类加载时创建实例,懒汉式则在首次调用时创建,后者在多线程环境下需使用同步机制保证线程安全。单例模式有助于提高代码的可维护性和扩展性,应根据需求选择合适实现方式。
25 1
|
15天前
|
SQL 设计模式 安全
Java编程中的单例模式深入解析
【8月更文挑战第27天】本文旨在探索Java中实现单例模式的多种方式,并分析其优缺点。我们将通过代码示例,展示如何在不同的场景下选择最合适的单例模式实现方法,以及如何避免常见的陷阱。
|
11天前
|
设计模式 安全 Java
Java编程中的单例模式深度解析
【8月更文挑战第31天】 单例模式,作为设计模式中的经典之一,在Java编程实践中扮演着重要的角色。本文将通过简洁易懂的语言,逐步引导读者理解单例模式的本质、实现方法及其在实际应用中的重要性。从基础概念出发,到代码示例,再到高级应用,我们将一起探索这一模式如何优雅地解决资源共享和性能优化的问题。
|
11天前
|
设计模式 安全 Java
Java中的单例模式:理解与实践
【8月更文挑战第31天】在软件设计中,单例模式是一种常用的设计模式,它确保一个类只有一个实例,并提供一个全局访问点。本文将深入探讨Java中实现单例模式的不同方法,包括懒汉式、饿汉式、双重校验锁以及静态内部类等方法。每种方法都有其适用场景和潜在问题,我们将通过代码示例来展示如何根据具体需求选择合适的实现方式。
|
11天前
|
设计模式 安全 Java
Java编程中的单例模式实现与应用
【8月更文挑战第31天】在Java的世界里,单例模式是构建高效且资源友好应用的基石之一。本文将深入浅出地介绍如何通过单例模式确保类只有一个实例,并提供一个全局访问点。我们将探索多种实现方法,包括懒汉式、饿汉式和双重校验锁,同时也会讨论单例模式在多线程环境下的表现。无论你是Java新手还是资深开发者,这篇文章都将为你打开一扇理解并有效应用单例模式的大门。
|
21天前
|
设计模式 SQL 缓存
Java编程中的设计模式:单例模式的深入理解与应用
【8月更文挑战第22天】 在Java的世界里,设计模式是构建可维护、可扩展和灵活的软件系统的基石。本文将深入浅出地探讨单例模式这一经典设计模式,揭示其背后的哲学思想,并通过实例演示如何在Java项目中有效运用。无论你是初学者还是资深开发者,这篇文章都将为你打开一扇洞悉软件设计深层逻辑的大门。
26 0
|
2月前
|
设计模式 安全 Java
Java面试题:设计模式如单例模式、工厂模式、观察者模式等在多线程环境下线程安全问题,Java内存模型定义了线程如何与内存交互,包括原子性、可见性、有序性,并发框架提供了更高层次的并发任务处理能力
Java面试题:设计模式如单例模式、工厂模式、观察者模式等在多线程环境下线程安全问题,Java内存模型定义了线程如何与内存交互,包括原子性、可见性、有序性,并发框架提供了更高层次的并发任务处理能力
60 1
|
1月前
|
设计模式 SQL 安全
单例模式大全:细说七种线程安全的Java单例实现,及数种打破单例的手段!
设计模式,这是编程中的灵魂,用好不同的设计模式,能使你的代码更优雅/健壮、维护性更强、灵活性更高,而众多设计模式中最出名、最广为人知的就是Singleton Pattern单例模式。通过单例模式,我们就可以避免由于多个实例的创建和销毁带来的额外开销,本文就来一起聊聊单例模式。