用C语言实现单例模式

单例模式的核心是确保一个类(或模块)只有一个实例,并提供全局访问点 。在C语言中,没有类的概念,但可以通过静态变量、函数封装实现类似效果,保证某块内存(如配置对象、管理器)仅被初始化一次。

C语言实现单例模式的关键思路

  1. 隐藏实例:用静态局部变量存储唯一实例(仅在首次访问时初始化)。
  2. 全局访问点:提供一个接口函数,负责初始化实例并返回其指针(确保仅初始化一次)。
  3. 线程安全(可选):在多线程环境下,需通过互斥锁避免并发初始化导致的多实例问题。

示例:配置管理器单例(单线程版)

假设需要一个全局唯一的配置管理器,负责加载和提供配置项(如数据库地址、端口)。

步骤1:定义单例结构体(ConfigManager)
c 复制代码
// 配置管理器结构体(单例的具体数据)
typedef struct {
    char db_host[64];   // 数据库地址
    int db_port;        // 数据库端口
    char log_path[128]; // 日志路径
} ConfigManager;
步骤2:实现单例的创建与访问接口

通过静态局部变量存储实例,并用接口函数控制初始化逻辑(仅首次调用时初始化)。

c 复制代码
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

// 静态局部变量:存储唯一实例(文件内可见,外部无法直接访问)
static ConfigManager* instance = NULL;

// 单例的全局访问接口
ConfigManager* ConfigManager_getInstance() {
    // 首次调用时初始化实例
    if (instance == NULL) {
        instance = (ConfigManager*)malloc(sizeof(ConfigManager));
        if (instance == NULL) {
            printf("内存分配失败\n");
            return NULL;
        }
        // 初始化配置(实际场景可能从文件读取)
        strcpy(instance->db_host, "localhost");
        instance->db_port = 3306;
        strcpy(instance->log_path, "/var/log/app.log");
        printf("配置管理器初始化完成\n");
    }
    return instance;
}

// 单例的销毁接口(可选,程序退出前调用)
void ConfigManager_destroy() {
    if (instance != NULL) {
        free(instance);
        instance = NULL;
        printf("配置管理器已销毁\n");
    }
}
步骤3:使用单例

无论调用多少次ConfigManager_getInstance,返回的都是同一个实例。

c 复制代码
int main() {
    // 第一次获取实例:触发初始化
    ConfigManager* config1 = ConfigManager_getInstance();
    printf("数据库地址: %s, 端口: %d\n", config1->db_host, config1->db_port);

    // 第二次获取实例:直接返回已存在的实例
    ConfigManager* config2 = ConfigManager_getInstance();
    printf("日志路径: %s\n", config2->log_path);

    // 验证是否为同一个实例(地址相同)
    if (config1 == config2) {
        printf("config1 和 config2 是同一个实例\n");
    }

    // 销毁单例(程序退出前)
    ConfigManager_destroy();
    return 0;
}

输出结果

复制代码
配置管理器初始化完成
数据库地址: localhost, 端口: 3306
日志路径: /var/log/app.log
config1 和 config2 是同一个实例
配置管理器已销毁

多线程安全的单例(加锁版)

在多线程环境中,多个线程可能同时进入instance == NULL的判断,导致创建多个实例。需通过互斥锁保证初始化的原子性。

c 复制代码
#include <pthread.h> // 需链接pthread库(编译时加 -lpthread)

// 静态实例
static ConfigManager* instance = NULL;
// 互斥锁:保证初始化过程唯一
static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

// 线程安全的单例访问接口
ConfigManager* ConfigManager_getInstance() {
    // 双重检查锁(Double-Checked Locking):减少锁开销
    if (instance == NULL) { // 第一次检查:避免每次加锁
        pthread_mutex_lock(&mutex); // 加锁
        if (instance == NULL) { // 第二次检查:确保仅初始化一次
            instance = (ConfigManager*)malloc(sizeof(ConfigManager));
            // ... 初始化配置 ...
            printf("线程安全的配置管理器初始化完成\n");
        }
        pthread_mutex_unlock(&mutex); // 解锁
    }
    return instance;
}

核心特点总结

  1. 唯一性 :通过静态变量instance确保全局只有一个实例。
  2. 延迟初始化:实例在首次访问时才创建,避免程序启动时的不必要开销。
  3. 封装性 :实例通过ConfigManager_getInstance接口访问,外部无法直接修改或创建,保证可控性。
  4. 线程安全(可选):多线程环境下需加锁,双重检查锁可平衡安全性和性能。

C语言的单例模式常用于全局资源管理(如配置、日志、设备驱动),核心是通过静态变量和函数封装控制实例的创建与访问。

相关推荐
Tingjct13 小时前
【初阶数据结构-二叉树】
c语言·开发语言·数据结构·算法
飞机和胖和黄14 小时前
考研之王道C语言第三周
c语言·数据结构·考研
醉颜凉14 小时前
【LeetCode】打家劫舍III
c语言·算法·leetcode·树 深度优先搜索·动态规划 二叉树
一匹电信狗14 小时前
【LeetCode_21】合并两个有序链表
c语言·开发语言·数据结构·c++·算法·leetcode·stl
卢锡荣17 小时前
Type-c OTG数据与充电如何进行交互使用应用讲解
c语言·开发语言·计算机外设·电脑·音视频
v_for_van18 小时前
力扣刷题记录2(无算法背景,纯C语言)
c语言·算法·leetcode
二年级程序员18 小时前
动态内存管理
c语言
我能坚持多久18 小时前
D20—C语言文件操作详解:从基础到高级应用
c语言·开发语言
(❁´◡`❁)Jimmy(❁´◡`❁)18 小时前
CF2188 C. Restricted Sorting
c语言·开发语言·算法
想放学的刺客19 小时前
单片机嵌入式试题(第27期)设计可移植、可配置的外设驱动框架的关键要点
c语言·stm32·单片机·嵌入式硬件·物联网