【不就是Java设计模式吗】解密单例八种设计模式,带你整理代码思路

简介: 【不就是Java设计模式吗】解密单例八种设计模式,带你整理代码思路

🧡前言

先看看思维导图:

image.png


📕在学习Java基础的时候,曾天真的以为单例设计模式就是懒汉式和饿汉式这两种。今天在B站学习了单例设计模式,真是打开了新世界的大门,自己总结了单例模式的学习笔记,从代码到自己的思路,最后到每一种方式的小结,相信小伙伴们看了后会有所收获


一、🍂模式介绍

所谓单例设计模式,就是采取一定的方法保证在整个的软件系统中,对某个类

只能存在一个对象实例,并且该类只提供一个取得其对象实例的方法(静态方法)


二、🍂分类

🏳‍🌈分类1:饿汉式

静态常量饿汉式

代码

public class Test01 {
    public static void main(String[] args) {
        Singleton instance1 = Singleton.getInstance();
        Singleton instance2 = Singleton.getInstance();
        //结果是true,说明是同一个对象实例
        System.out.println(instance1 == instance2);
    }
}
class Singleton {
    //构造器私有化,外部不能通过new的方式创建此类的对象
    private Singleton() {}
    //本类内部创建对象实例
    private final static Singleton instance = new Singleton();
    //提供静态方法,得到实例对象
    public static Singleton getInstance() {
        return instance;
    }
}


🔥思路

🚩先创建私有构造器,然后创建私有的final的静态对象,最后提供一个可以得到实例对象的方法


结论与分析

线程安全,写法简单,该方式可用,在类装载的时候就完成了实例化。也就是因为在类装载的时候完成了实例化,所以,如果从始至终都从未使用过这个实例,就造成了内存的浪费


静态代码块饿汉式

代码

class Singleton {
    private Singleton() {}
    private static Singleton instance;
    static {
        instance = new Singleton();
    }
    public static Singleton getInstance() {
        return instance;
    }
}


🔥思路

🚩先创建私有类构造器,然后创建私有静态类引用,接着在静态代码块中给引用创建实例,最后提供一个可以得到实例对象的方法


结论与分析

线程安全,写法简单,该方式可用,在类装载的时候就完成了实例化。也就是因为在类装载的时候完成了实例化,所以,如果从始至终都从未使用过这个实例,就造成了内存的浪费


🏳‍🌈分类2:懒汉式

方式1

代码

class Singleton{
    private Singleton(){}
    private static Singleton instance;
    //当使用该方法时,才去创建对象
    public static Singleton getInstance(){
        if (instance == null)
            instance = new Singleton();
        return instance;
    }
}


🔥思路

🚩跟前面的饿汉式思路差不多,先创建一个私有类构造器,然后提供私有静态引用,最后写一个静态方法用来得到引用的对象实例;其中,引用的对象实例是用if (instance == null)来判断是否赋值的


结论与分析

线程不安全,当在多线程下,一个线程进入了if (instance == null)判断语句,还未来得及往下执行,另一个线程也通过了这个判断语句,这时候就可能产生多个实例,不建议使用


方式2

代码

class Singleton{
    private static Singleton instance;
    private Singleton(){}
    //加入同步处理的代码,解决线程安全问题
    public static synchronized Singleton getInstance(){
        if (instance == null)
            instance  = new Singleton();
        return instance;
    }
}


🔥思路

🚩思路跟方式1就是在方法中多了一个synchronized,可以保证线程安全


结论与分析

线程安全,但是效率低。每个线程在想获得类的实例的时候,执行getInstance()都要进行同步,其实这个方法只执行一次实例化代码就可以了,不推荐使用


方式3

代码

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


🔥思路

🚩跟前面的代码没有很大区别,不同的是,方式3把synchronized线程同步问题放在getInstance()里面,但是并没有解决安全问题


结论与分析

线程不安全,效率也低,不推荐使用


🏳‍🌈分类3:双重检查

代码

class Singleton{
 /*
    volatile的作用:
    1、保证了不同线程对该变量操作的内存可见性(当一个线程修改了变量,其他使用次变量的线程可以立即知道这一修改)
    2、禁止了指令重排序.
     */
    private static volatile Singleton instance;
    private Singleton(){}
    public static Singleton getInstance(){
        if (instance == null){
            synchronized(Singleton.class){
                if (instance == null)
                    instance = new Singleton();
            }
        }
        return instance;
    }
}



🔥思路

🚩这里可以和懒汉式的方式3对比一下,大体上比方式3多了一层if语句的判断


总结分析

线程安全,效率较高;如代码所示,进行了两次if (instance == null)检查,可以保证线程安全,推荐使用


🏳‍🌈分类4:静态内部类

代码

class Singleton{
    private Singleton(){}
    private static class SingletonInstance{
        private static final Singleton INSTANCE = new Singleton();
    }
    public static Singleton getInstance(){
        return SingletonInstance.INSTANCE;
    }
}


🔥思路

🚩这个思路也比较好理解,先构造私有类构造器方法,然后创建静态内部类用来创建类的对象,最后创建一个从内部类获取对象实例的方法


总结分析

线程安全,采用了类装载的机制来保证初始化实例时只有一个线程,效率高;静态内部类方式在Singleton类被装载时并不会立即实例化,而是在需要实例化时,调用getInstance方法,才会装载SingletonInstance类,从而完成Singleton的实例化,推荐使用


🏳‍🌈分类5:枚举

代码

public class Test08 {
    public static void main(String[] args) {
        Singleton instance1 = Singleton.INSTANCE;
        Singleton instance2 = Singleton.INSTANCE;
        //true
        System.out.println(instance1 == instance2);
        //运行结果:你好呀,我是卷心菜~~
        instance1.sayHello();
    }
}
enum Singleton{
    INSTANCE;
    public void sayHello(){
        System.out.println("你好呀,我是卷心菜~~");
    }
}



🔥思路

🚩枚举类真的是单例模式中最简单的一个了,就是根据自己需要的来写相应的属性和方法


总结分析

线程安全,能防止反序列化重新创建新的对象,非常推荐使用


三、🍂使用场景

🚩需要频繁的进行创建和销毁的对象、创建对象时耗时过多或耗费资源过多(即:重量级对象),但又经常用到的对象、工具类对象、频繁访问数据库或文件的对象(比如数据源、session工厂等)


四、🍂总结

单例模式保证了 系统内存中该类只存在一个对象,节省了系统资源,对于一些需要频繁创建销毁的对象,使用单例模式可以提高系统性能

当想实例化一个单例类的时候,必须要记住使用相应的获取对象的方法,而不是使用new来获取对象


相关文章
|
7天前
|
设计模式 Java 关系型数据库
【Java笔记+踩坑汇总】Java基础+JavaWeb+SSM+SpringBoot+SpringCloud+瑞吉外卖/谷粒商城/学成在线+设计模式+面试题汇总+性能调优/架构设计+源码解析
本文是“Java学习路线”专栏的导航文章,目标是为Java初学者和初中高级工程师提供一套完整的Java学习路线。
|
3天前
|
设计模式 Java
Java设计模式:组合模式的介绍及代码演示
组合模式是一种结构型设计模式,用于将多个对象组织成树形结构,并统一处理所有对象。例如,统计公司总人数时,可先统计各部门人数再求和。该模式包括一个通用接口、表示节点的类及其实现类。通过树形结构和节点的通用方法,组合模式使程序更易扩展和维护。
Java设计模式:组合模式的介绍及代码演示
|
7天前
|
设计模式 安全 算法
【Java面试题汇总】设计模式篇(2023版)
谈谈你对设计模式的理解、七大原则、单例模式、工厂模式、代理模式、模板模式、观察者模式、JDK中用到的设计模式、Spring中用到的设计模式
【Java面试题汇总】设计模式篇(2023版)
|
7天前
|
设计模式 Java 关系型数据库
【Java笔记+踩坑】设计模式——原型模式
对比原型模式和传统方式的实现思路、代码方案、优缺点,阐述原型模式的使用场景,以及深拷贝、浅拷贝等相关概念,并扩展原型模式在Spring源码中的应用。
【Java笔记+踩坑】设计模式——原型模式
|
7天前
|
设计模式 存储 安全
设计模式——设计模式介绍和单例设计模式
饿汉式(静态常量)、饿汉式(静态代码块)、懒汉式(线程不安全)、懒汉式(线程安全,同步方法)、懒汉式(线程不安全,同步代码块)、双重检查(推荐,线程安全、懒加载)、静态内部类(推荐)、枚举(推荐)
设计模式——设计模式介绍和单例设计模式
|
3天前
|
Java 程序员 API
Java中的Lambda表达式:简化代码的秘密武器
在Java 8中引入的Lambda表达式是一种强大的编程工具,它可以显著简化代码,提高可读性。本文将介绍Lambda表达式的基本概念、优势以及在实际开发中的应用。通过具体示例,您将了解如何使用Lambda表达式来简化集合操作、线程编程和函数式编程。让我们一起探索这一革命性的特性,看看它是如何改变Java编程方式的。
16 4
|
3天前
|
Java 开发者
探索Java中的Lambda表达式:简化你的代码
【8月更文挑战第49天】在Java 8的发布中,Lambda表达式无疑是最令人兴奋的新特性之一。它不仅为Java开发者提供了一种更加简洁、灵活的编程方式,而且还极大地提高了代码的可读性和开发效率。本文将通过实际代码示例,展示如何利用Lambda表达式优化和重构Java代码,让你的编程之旅更加轻松愉快。
|
8天前
|
SQL JavaScript 前端开发
基于Java访问Hive的JUnit5测试代码实现
根据《用Java、Python来开发Hive应用》一文,建立了使用Java、来开发Hive应用的方法,产生的代码如下
28 6
|
5天前
|
设计模式 数据库连接 PHP
PHP中的设计模式:提高代码的可维护性与扩展性
在本文中,我们将深入探讨PHP编程语言中设计模式的重要性,以及如何通过应用这些模式来提高代码的可维护性和扩展性。设计模式是一套被反复使用、经过分类编目的代码设计经验的总结。它们代表了最佳的实践,能帮助开发者编写出更加健壮、灵活和可复用的代码。本文将介绍几种常见的设计模式,并通过PHP代码示例展示其应用。
|
6天前
|
Java 开发者
探索Java中的Lambda表达式:简化代码,提升效率
【9月更文挑战第14天】本文旨在揭示Java 8中引入的Lambda表达式如何革新了我们编写和管理代码的方式。通过简洁明了的语言和直观的代码示例,我们将一起走进Lambda表达式的世界,了解其基本概念、语法结构以及在实际编程中的应用。文章不仅会展示Lambda表达式的魅力所在,还会指导读者如何在日常工作中有效利用这一特性,以提高编码效率和程序可读性。