广告
返回顶部
首页 > 资讯 > 后端开发 > Python >详解Java单例模式的实现与原理剖析
  • 562
分享到

详解Java单例模式的实现与原理剖析

2024-04-02 19:04:59 562人浏览 安东尼

Python 官方文档:入门教程 => 点击学习

摘要

目录一、什么是单例模式二、哪些地方用到了单例模式三、单例模式的优缺点优点缺点四、手写单例模式饿汉式枚举饿汉式DCL懒汉式双检锁懒汉式内部类懒汉式小结一、什么是单例模式 单例模式(Si

一、什么是单例模式

单例模式(Singleton Pattern)是 Java 中最简单的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。

这种模式涉及到一个单一的类,该类负责创建自己的对象,同时确保只有单个对象被创建。这个类提供了一种访问其唯一的对象的方式,可以直接访问,不需要实例化该类的对象。

注意

  • 单例模式只能由一个实例对象
  • 单例类必须自己创建自己的唯一实例。
  • 单例类必须给所有其他对象提供这一实例。

二、哪些地方用到了单例模式

单例模式经常用在需要一个实例的程序中,例如

1.spring框架ioc容器就使用到了单例模式,默认创建对象的时候为单例模式

2.ResultBean 后端统一返回给前端的封装类,这个在项目中是唯一的,只用一个对象进行返回JSON给前端进行渲染

jdk中也有单例模式的身影,例

  • Runtime 体现了饿汉式单例
  • Console 体现了双检锁懒汉式单例
  • Collections 中的 EmptyNavigableSet 内部类懒汉式单例
  • ReverseComparator.REVERSE_ORDER 内部类懒汉式单例
  • Comparators.NaturalOrderComparator.INSTANCE 枚举饿汉式单例

三、单例模式的优缺点

优点

提供了对唯一实例的访问

可以节约系统资源,提高系统的性能,减少不必要的内存开销

允许可变数目的实例(多例类)

缺点

扩展困难(缺少抽象层)

单例类的职责过重

由于自动垃圾回收机制,可能会导致共享的单例对象的状态丢失

四、手写单例模式

饿汉式

package com.wanshi.single;

//饿汉式单例
public class Hungry {


    //会造成资源浪费,占用CPU
    private byte[] data1 = new byte[1024*1024];
    private byte[] data2 = new byte[1024*1024];
    private byte[] data3 = new byte[1024*1024];
    private byte[] data4 = new byte[1024*1024];


    private Hungry() {
        System.out.println("Hungry init...");
    }


    private static Hungry hungry = new Hungry();

    public static Hungry getInstance() {
        return hungry;
    }
}

class Test {
	public static void main(String[] args) {
        Hungry hungry = Hungry.getInstance();
        Hungry hungry2 = Hungry.getInstance();
        System.out.println(hungry);
        System.out.println(hungry2);
    }
}

枚举饿汉式

package com.wanshi.single;

import java.lang.reflect.Constructor;

// enum 是一个class类
public enum EnumSingle {

    INSTANCE;

    public static EnumSingle getInstance() {
        return INSTANCE;
    }
}

class Test {

    public static void main(String[] args) throws Exception{
        EnumSingle instance1 = EnumSingle.INSTANCE;
        Constructor<EnumSingle> declaredConstructor = EnumSingle.class.getDeclaredConstructor(String.class, int.class);
        declaredConstructor.setAccessible(true);

        EnumSingle instance2 = declaredConstructor.newInstance();

        System.out.println(instance1);
        System.out.println(instance2);
    }
}

在这里自行下载jad编译工具即可

枚举类最后反编译源码

jad工具反编译

jad -sjava EnumSingle.class

// Decompiled by Jad v1.5.8g. Copyright 2001 Pavel Kouznetsov.
// Jad home page: Http://www.kpdus.com/jad.html
// Decompiler options: packimports(3) 
// Source File Name:   EnumSingle.java

package com.wanshi.single;


public final class EnumSingle extends Enum
{

    public static EnumSingle[] values()
    {
        return (EnumSingle[])$VALUES.clone();
    }

    public static EnumSingle valueOf(String name)
    {
        return (EnumSingle)Enum.valueOf(com/wanshi/single/EnumSingle, name);
    }

    private EnumSingle(String s, int i)
    {
        super(s, i);
    }

    public static EnumSingle getInstance()
    {
        return INSTANCE;
    }

    public static final EnumSingle INSTANCE;
    private static final EnumSingle $VALUES[];

    static 
    {
        INSTANCE = new EnumSingle("INSTANCE", 0);
        $VALUES = (new EnumSingle[] {
            INSTANCE
        });
    }
}

DCL懒汉式

package com.wanshi.single;

public class Lazy {

    private static Lazy lazy;

    public static Lazy getInterface() {
        synchronized (Lazy.class) {
            if (lazy == null) {
                lazy = new Lazy();
            }
        }
        return lazy;
    }

    public static void main(String[] args) {
        Lazy lazy = Lazy.getInterface();
        Lazy lazy2 = Lazy.getInterface();
        System.out.println(lazy);
        System.out.println(lazy2);
    }
}

双检锁懒汉式

package com.wanshi.single;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;

public class LazyMan {


    private static boolean flag = false;

    private LazyMan() {
        synchronized (this) {
            if (!flag) {
                flag = true;
            } else {
                throw new RuntimeException("不要试图通过反射破坏对象");
            }
        }
    }

    private volatile static LazyMan lazyMan;

    //双重检查锁,懒汉式(DCL懒汉式)
    public static LazyMan getInstance() {
        if (lazyMan == null) {
            synchronized (LazyMan.class) {
                if (lazyMan == null) {
                    //不是原子性操作,1.分配内存空间,2.执行构造方法,3.把对象指向这个空间 指令重排可能会发生   加上volatile关闭指令重排
                    lazyMan = new LazyMan();
                }
            }
        }
        return lazyMan;
    }

    public static void main(String[] args) throws Exception {
//        LazyMan lazyMan1 = LazyMan.getInstance();

        Field flag = LazyMan.class.getDeclaredField("flag");
        flag.setAccessible(true);


        Constructor<LazyMan> declaredConstructor = LazyMan.class.getDeclaredConstructor(null);

        declaredConstructor.setAccessible(true);
        LazyMan lazyMan1 = declaredConstructor.newInstance();
        flag.set(lazyMan1, false);
        LazyMan lazyMan2 = declaredConstructor.newInstance();
        System.out.println(lazyMan1);
        System.out.println(lazyMan2);
    }
}

为什么要使用 volatile 关键字呢

不是原子性操作

1.分配内存空间

2.执行构造方法

3.把对象指向这个空间

指令重排可能会发生 加上volatile关闭指令重排

内部类懒汉式

package com.wanshi.single;

public class Holder {

    private Holder() {

    }

    public static class InnerClass {
        private static final Holder HOLDER = new Holder();
    }
}

案例全部通过测试

小结

单例模式共有5种创建方式,分别为饿汉式、DCL懒汉式、双检锁懒汉式、Enum枚举饿汉式,内部类懒汉式,这几种方式要掌握,项目中对于全局唯一的对象将其封装为单例模式,开箱即用,非常方便,以及面试中,会让手写单例模式,可谓是大厂必备!

以上就是详解Java单例模式的实现与原理剖析的详细内容,更多关于Java单例模式的资料请关注编程网其它相关文章!

--结束END--

本文标题: 详解Java单例模式的实现与原理剖析

本文链接: https://www.lsjlt.com/news/148523.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 详解Java单例模式的实现与原理剖析
    目录一、什么是单例模式二、哪些地方用到了单例模式三、单例模式的优缺点优点缺点四、手写单例模式饿汉式枚举饿汉式DCL懒汉式双检锁懒汉式内部类懒汉式小结一、什么是单例模式 单例模式(Si...
    99+
    2022-11-13
  • Java单例模式与破坏单例模式概念原理深入讲解
    目录什么是单例模式饿汉式(预加载)懒汉式(懒加载)反射破坏单例模式什么是单例模式 经典设计模式又分23种,也就是GoF 23 总体分为三大类: 创建型模式结构性模式行为型模式 Jav...
    99+
    2023-02-21
    Java单例模式 Java破坏单例模式
  • java 单例模式的实例详解
    java 单例模式的实例详解概念:    java中单例模式是一种常见的设计模式,单例模式分三种:懒汉式单例、饿汉式单例、登记式单例三种。    单例模式有一下特点:   1、单例类只能有一个实例。   2、单例类必须自己自己创建自己的唯一...
    99+
    2023-05-31
    java 单例模式 ava
  • Android 断点续传的原理剖析与实例讲解
     本文所要讲的是Android断点续传的内容,以实例的形式进行了详细介绍。        一、断点续传的原...
    99+
    2022-06-06
    断点续传 断点 Android
  • Java单例模式的6种实现方式详解
    目录为什么使用单例模式使用单例模式需要注意的关键点单例模式的几种写法1. 饿汉式2. 懒汉式3. DCL(Double CheckLock)实现单例4. 静态内部类5...
    99+
    2022-11-12
  • java设计模式-单例模式实现方法详解
    目录饿汉式静态变量静态代码块懒汉式线程不安全线程安全双重检查静态内部类总结单例模式,属于创建类型的一种常用的软件设计模式。通过单例模式的方法创建的类在当前进程中只有一个实例(根据需要...
    99+
    2022-11-12
  • Java设计模式之单例模式实例详解【懒汉式与饿汉式】
    本文实例讲述了Java设计模式之单例模式。分享给大家供大家参考,具体如下:单例模式就是产生一个对象实例,供外外部访问。它的应用场景就是在这个类在全局真资源需要统一访问,否则会造成混乱时,才有必要设计成单例。懒汉式,就是在使用这个对象时,才去...
    99+
    2023-05-31
    java 设计模式 单例模式
  • Java之单例模式实现方案详解
      单例模式是最常用到的设计模式之一,熟悉设计模式的朋友对单例模式都不会陌生。一般介绍单例模式的书籍都会提到 饿汉式 和 懒汉式 这两种实现方式。但是除了这两种方式,本文还会介绍其他...
    99+
    2022-11-12
  • JAVA 枚举单例模式及源码分析的实例详解
    JAVA 枚举单例模式及源码分析的实例详解      单例模式的实现有很多种,网上也分析了如今实现单利模式最好用枚举,好处不外乎三点:线程安全不会因为序列化而产生新实例防止反射攻击但是貌似没...
    99+
    2023-05-31
    java 枚举 单例
  • JavaScript单例模式能不能去实例只留单原理解析
    目录一、单例模式的分类二、惰性单例模式二、普通单例模式总结一、单例模式的分类 一个环境中有且只有一个实例,并且当前环境可以访问到它。往小了说,当前环境可以是一个函数作用域、块级作用...
    99+
    2022-12-22
    JavaScript单例模式去实例 JavaScript单例模式
  • Python实现单例模式的四种方式详解
    简介:单例模式可以保证一个类仅有一个实例,并提供一个访问它的全局访问点。适用性于当类只能有一个实例而且客户可以从一个众所周知的访问点访问它,例如访问数据库、MQ等。 实现方式: 1、...
    99+
    2022-11-11
  • Java单例模式的攻击与防御怎么理解
    这篇文章主要介绍“Java单例模式的攻击与防御怎么理解”,在日常操作中,相信很多人在Java单例模式的攻击与防御怎么理解问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java单例模式的攻击与防御怎么理解”的疑...
    99+
    2023-06-02
  • Golang sync.Once实现单例模式的方法详解
    目录1. sync.Once 的原理和实现2. sync.Once 的错误处理3. sync.Once 的嵌套调用4. 并发性能5. 总结Go 语言的 sync 包提供了一系列同步原...
    99+
    2023-05-18
    Golang sync.Once实现单例模式 Golang sync.Once原理 Golang sync.Once使用 Golang sync.Once
  • C++深入详解单例模式与特殊类设计的实现
    目录单例模式什么是单例模式应用场景优缺点实现饿汉模式懒汉模式特殊类设计设计一个类只能在堆上创建对象方法一方法二只能在栈上创建对象方法一方法二一个类不能被继承最后单例模式 什么是单例模...
    99+
    2022-11-13
  • Java的单例模式与final及抽象类和接口实例分析
    这篇文章主要介绍“Java的单例模式与final及抽象类和接口实例分析”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Java的单例模式与final及抽象类和接口实例分析”文章能帮助大家解决问题。1....
    99+
    2023-06-30
  • C++单例模式的几种实现方法详解
    目录局部静态变量方式静态成员变量指针方式智能指针方式辅助类智能指针单例模式通用的单例模板类总结局部静态变量方式 //通过静态成员变量实现单例 //懒汉式 class Single2 ...
    99+
    2022-11-13
  • 分析java中全面的单例模式多种实现方式
    目录一、单例模式的思想二、单例模式的 N 种实现方式2.1、饿汉式(线程安全,可用)2.2、常量式(线程安全,可用)2.3、懒汉式(线程不安全,并发场景不可用)2.4、同步的懒汉式?...
    99+
    2022-11-12
  • 详解Redis分布式锁的原理与实现
    目录前言使用场景为什么要使用分布式锁如何使用分布式锁流程图分布式锁的状态分布式锁的特点分布式锁的实现方式(以redis分布式锁实现为例)总结前言 在单体应用中,如果我们对共享数据不进行加锁操作,会出现数据一致性问题,我们...
    99+
    2022-06-23
    Redis分布式锁使用 Redis 分布式锁
  • Java中Prime算法的原理与实现详解
    目录Prim算法介绍1.点睛2.算法介绍3. 算法步骤4.图解Prime 算法实现1.构建后的图2.代码3.测试Prim算法介绍 1.点睛 在生成树的过程中,把已经在生成树中的节点看...
    99+
    2022-11-13
  • 详解Java ReentrantReadWriteLock读写锁的原理与实现
    目录概述原理概述加锁原理图解过程源码解析解锁原理图解过程源码解析概述 ReentrantReadWriteLock读写锁是使用AQS的集大成者,用了独占模式和共享模式。本文和大家一起...
    99+
    2022-11-13
    Java ReentrantReadWriteLock读写锁 Java ReentrantReadWriteLock Java 读写锁
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作