当前位置: 首页 > article >正文

【再谈设计模式】单例模式~唯一性的守护者

一、引言

        在软件工程中,软件开发,设计模式是提高代码复用性和可维护性的有效工具。单例模式(Singleton Pattern)作为一种创建型设计模式,旨在确保一个类只有一个实例,并提供对该实例的全局访问。这一模式在许多场景中都显得尤为重要,尤其是在需要共享资源或协调系统状态时。

二、定义与描述

        单例模式的核心思想是限制一个类仅能实例化一次,并提供一个全局访问点。这样可以避免重复实例化导致的资源浪费和状态不一致的问题。

结构

单例模式通常包含以下几个组成部分:

  • 私有构造函数:防止外部类直接实例化。
  • 一个静态变量:持有唯一的实例。
  • 一个静态方法:用于获取该实例。

三、抽象背景

        单例模式的背景可以追溯到多个设计需求的实际应用。例如,当一个应用程序需要使用一个共享的配置文件、数据库连接或共享资源时,单例模式便成为一个很好的选择。在这种情况下,程序不能允许多个实例同时存在,以免造成数据污染和不一致。

四、适用场景与现实问题解决

单例模式常用于以下场景:

  • 配置管理:在整个应用中共享配置文件或参数。
  • 日志管理:全局共享的日志记录器。
  • 数据库连接池:管理数据库连接的共享实例。

  • 线程池管理:确保线程池在整个应用中只有一个实例。

       通过应用单例模式,可以有效避免各类资源的误用和浪费,从而提升程序的性能和稳定性。旨在确保一个类只有一个实例,并提供一个全局访问点。为了让这个概念更易于理解,我们可以使用一个生活中的实际问题来说明。

生活例子:办公室的打印机

        在一个办公室工作,办公室里有一台打印机。现在我们来看看这台打印机在使用中的几种情况:

        唯一性:这个办公室只能有一台打印机。想象一下,如果有多台打印机,可能会导致很多问题,比如文件打印到错误的打印机、打印任务混乱等。因此,我们需要确保只有一台打印机。

        共享访问:办公室里的每个人都需要使用这台打印机,无论是打印文件还是复印资料。因此,我们需要一个简单的方法让每个人都能访问这台打印机,而不需要每次都去找。

        节省资源:维护多台打印机的成本和管理会耗费大量时间和精力。只有一台打印机可以集中管理,更加高效。

单例模式的类比

        在编程中,单例模式就像这台办公室的打印机一样。它确保一个类(例如打印机类)只有一个实例,并提供一个全局的访问点。在代码中实现单例模式通常涉及以下几个步骤:

  • 私有构造函数:防止其他地方直接创建新实例。
  • 静态变量:用于保存唯一的实例。
  • 公共静态方法:提供访问该实例的方式。

        通过这种方式,代码中需要使用打印机的地方,都会获取到同一个实例,避免了实例重复,确保了全局统一性。

        单例模式就像是确保办公室只有一台打印机,避免了资源的浪费和管理的复杂性,让每个人都可以方便地访问这台打印机。而在实际编程中,单例模式则确保了一个类的唯一性和资源的高效利用。

五、初衷与问题解决

        单例模式的初衷是为了控制对象的数量,使其不被无意创建。许多开发者面临对象管理不当导致的资源浪费和状态不一致的问题,而这一模式恰好能够有效解决这些问题。单例模式实现了对全局资源的集中管理,提高了系统的可维护性和安全性。

六、各语言实现与编码示例

Java 实现

public class Singleton {
    private static Singleton instance;

    private Singleton() {}

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

Go 实现

package main

import "sync"

var instance *Singleton
var once sync.Once

type Singleton struct{}

func GetInstance() *Singleton {
    once.Do(func() {
        instance = &Singleton{}
    })
    return instance
}

C++ 实现

class Singleton {
private:
    static Singleton* instance;
    Singleton() {}
public:
    static Singleton* getInstance() {
        if (instance == nullptr) {
            instance = new Singleton();
        }
        return instance;
    }
};

Singleton* Singleton::instance = nullptr;

Python 实现

class Singleton:
    _instance = None

    def __new__(cls, *args, **kwargs):
        if not cls._instance:
            cls._instance = super(Singleton, cls).__new__(cls, *args, **kwargs)
        return cls._instance
    
singleton1 = Singleton()
singleton2 = Singleton()

assert singleton1 is singleton2  # True

七、单例模式的优缺点

优点

  • 节省内存:同一时刻只存在一个实例。
  • 全局访问:提供全局访问点,方便在程序的不同部分共享资源。
  • 控制资源使用:通过限制实例化次数,可以避免资源浪费。

缺点

  • 可扩展性差:在某些场景下可能会导致系统中的“单点故障”。
  • 更难测试:单例模式可能使得测试变得复杂,因为它引入了全局状态。
  • 潜在的线程安全问题:在多线程环境中需谨慎处理,可能需要额外的同步机制。

八、单例模式的升级版

        在实际应用中,单例模式可能会演变成更复杂的形式,包括饿汉单例和懒汉单例。

饿汉单例

        饿汉单例在类被加载时就创建实例,因此实例的创建是提前的。通常使用一个静态常量来存储单例的实例。

特点

  • 线程安全:实例在类加载时就创建,因此不需要额外的同步机制。
  • 容易实现:比较简单,没有复杂的逻辑。
  • 资源浪费:即使没有使用,也会在加载时就创建实例,这可能导致资源浪费。

示例代码

public class HungrySingleton {
    // 在静态变量中创建唯一的实例
    private static final HungrySingleton INSTANCE = new HungrySingleton();

    // 私有构造函数,防止外部创建对象
    private HungrySingleton() {}

    // 提供一个全局访问点
    public static HungrySingleton getInstance() {
        return INSTANCE;
    }
}

懒汉单例

        懒汉单例在第一次被调用时才会创建实例,因此实例的创建是延迟的。通常会使用一个私有静态变量来存储实例,并在访问时检查是否已经被创建。

特点

  • 延迟加载:只有在需要的时候才创建实例,如果不需要,则不会创建。
  • 线程不安全:如果多个线程同时访问 getInstance() 方法,可能会导致创建多个实例。为了解决这个问题,常常在方法上添加同步机制。
  • 实现复杂:需要考虑多线程访问的安全性和性能。

示例代码(线程不安全):

public class LazySingleton {
    private static LazySingleton instance;

    private LazySingleton() {}

    public static LazySingleton getInstance() {
        if (instance == null) {
            instance = new LazySingleton();
        }
        return instance;
    }
}

示例代码(线程安全,使用同步):

public class ThreadSafeLazySingleton {
    private static ThreadSafeLazySingleton instance;

    private ThreadSafeLazySingleton() {}

    public static synchronized ThreadSafeLazySingleton getInstance() {
        if (instance == null) {
            instance = new ThreadSafeLazySingleton();
        }
        return instance;
    }
}
  • 饿汉单例适合于实例创建比较简单,并且应用程序总是需要这个实例的情况。
  • 懒汉单例适合于实例创建开销比较大,或者在特定条件下才需要创建的情况,但需要注意线程安全的问题。

http://www.kler.cn/a/372311.html

相关文章:

  • 【Nas】X-DOC:Mac mini Docker部署小雅Alist
  • sql进阶篇
  • 开源项目-投票管理系统
  • Pytorch(一)
  • 【Cri-Dockerd】安装cri-dockerd
  • 被上传文件于后端的命名策略
  • Dockerfile制作Oracle19c镜像
  • xpath爬虫
  • 多线程显示 CSV 2 PNG 倒计时循环播放
  • 低功耗模组学习指南:从入门到精通通过MQTT连接实现远程控制
  • 如何在不同设备上轻松下载Facebook应用:全面指南
  • AI助力医疗数据自动化:诊断报告识别与管理
  • TCP全连接队列与 tcpdump 抓包
  • vue点击菜单,出现2个相同tab,啥原因
  • 代码备份管理 —— Git实用操作
  • Spring Boot框架下的酒店住宿登记系统
  • Centos如何卸载docker
  • WPF中视觉树和逻辑树的区别和联系
  • HTML入门教程2:HTML发展历史
  • 分布式数据库技术金融应用规范技术架构
  • java 中 List<T> 类型数据在 postgreSql 数据库中存储
  • 有效沟通与系统思考
  • 数据结构和算法-动态规划(3)-经典问题
  • Springboot整合RocketMQ分布式事务
  • 博科交换机SNMP采集(光衰)信息
  • 【Hive复杂数据类型和函数】全网总结最全的Hive函数