用C语言实现适配器模式

适配器模式(Adapter Pattern)的核心是将一个接口转换为客户端期望的另一个接口 ,使原本因接口不兼容而无法协作的类(或模块)能够一起工作。在C语言中,可通过封装不兼容的接口、提供统一的适配接口来实现。

C语言实现适配器模式的思路

  1. 目标接口(Target):客户端期望的统一接口(函数指针结构体)。
  2. 适配者(Adaptee):已存在的、接口不兼容的模块(如旧系统函数、第三方库)。
  3. 适配器(Adapter):实现目标接口,内部调用适配者的接口,完成接口转换。

示例:适配不同格式的日志库

假设客户端期望一个统一的日志接口(log_infolog_error),但现有两个日志库的接口不兼容:

  • 旧日志库:old_log(message, level)(级别用整数表示:1=INFO,2=ERROR)。
  • 新日志库:new_log(level, message)(级别用字符串表示:"INFO"、"ERROR")。
步骤1:定义目标接口(客户端期望的接口)
c 复制代码
// 目标接口:客户端期望的统一日志接口
typedef struct {
    void (*log_info)(const char* message);  // 打印INFO级日志
    void (*log_error)(const char* message); // 打印ERROR级日志
} LogTarget;
步骤2:定义适配者(不兼容的日志库)
c 复制代码
#include <stdio.h>

// 适配者1:旧日志库(接口:message在前,级别是整数)
void old_log(const char* message, int level) {
    const char* level_str = (level == 1) ? "INFO" : "ERROR";
    printf("[旧日志] %s: %s\n", level_str, message);
}

// 适配者2:新日志库(接口:级别在前,级别是字符串)
void new_log(const char* level, const char* message) {
    printf("[新日志] %s: %s\n", level, message);
}
步骤3:实现适配器(将适配者接口转换为目标接口)
3.1 旧日志库适配器
c 复制代码
// 旧日志库的适配器:实现LogTarget接口
static void old_adapter_log_info(const char* message) {
    // 调用旧日志库,转换参数(INFO对应级别1)
    old_log(message, 1);
}

static void old_adapter_log_error(const char* message) {
    // 调用旧日志库,转换参数(ERROR对应级别2)
    old_log(message, 2);
}

// 初始化旧日志适配器
LogTarget* old_log_adapter_create() {
    static LogTarget adapter; // 静态实例,避免动态分配
    adapter.log_info = old_adapter_log_info;
    adapter.log_error = old_adapter_log_error;
    return &adapter;
}
3.2 新日志库适配器
c 复制代码
// 新日志库的适配器:实现LogTarget接口
static void new_adapter_log_info(const char* message) {
    // 调用新日志库,转换参数(级别字符串"INFO")
    new_log("INFO", message);
}

static void new_adapter_log_error(const char* message) {
    // 调用新日志库,转换参数(级别字符串"ERROR")
    new_log("ERROR", message);
}

// 初始化新日志适配器
LogTarget* new_log_adapter_create() {
    static LogTarget adapter;
    adapter.log_info = new_adapter_log_info;
    adapter.log_error = new_adapter_log_error;
    return &adapter;
}
步骤4:客户端使用(依赖目标接口,无需关心具体适配者)
c 复制代码
// 客户端函数:仅依赖目标接口,不直接调用旧/新日志库
void client_code(LogTarget* logger) {
    logger->log_info("系统启动成功");
    logger->log_error("配置文件不存在");
}

int main() {
    // 使用旧日志库(通过适配器)
    LogTarget* old_logger = old_log_adapter_create();
    printf("=== 使用旧日志库 ===\n");
    client_code(old_logger);

    // 使用新日志库(通过适配器)
    LogTarget* new_logger = new_log_adapter_create();
    printf("\n=== 使用新日志库 ===\n");
    client_code(new_logger);

    return 0;
}

输出结果

复制代码
=== 使用旧日志库 ===
[旧日志] INFO: 系统启动成功
[旧日志] ERROR: 配置文件不存在

=== 使用新日志库 ===
[新日志] INFO: 系统启动成功
[新日志] ERROR: 配置文件不存在

核心思想总结

  1. 接口转换 :适配器通过封装适配者的接口(如old_lognew_log),将其转换为客户端期望的目标接口(LogTarget),屏蔽了接口差异。
  2. 解耦客户端与适配者 :客户端仅依赖LogTarget接口,无需修改即可切换不同的日志库(旧→新),符合开放-封闭原则
  3. 复用现有模块:无需修改旧日志库或新日志库的代码,通过适配器使其能在新系统中复用。

扩展:对象适配器 vs 类适配器

  • 对象适配器 (本示例):通过在适配器中调用适配者的函数(如old_log)实现,更灵活(可动态切换适配者)。
  • 类适配器(C语言模拟):若适配者是结构体,可让适配器结构体"继承"适配者结构体,直接复用其成员(类似面向对象的继承)。

C语言的适配器模式常用于整合旧系统、第三方库,或统一不同模块的接口,是系统兼容性设计的重要手段。

相关推荐
superman超哥1 天前
仓颉语言中元组的使用:深度剖析与工程实践
c语言·开发语言·c++·python·仓颉
charlie1145141911 天前
现代嵌入式C++教程:C++98——从C向C++的演化(2)
c语言·开发语言·c++·学习·嵌入式·教程·现代c++
雨季余静1 天前
c语言 gb2312转utf-8,带码表,直接使用。
c语言·c语言utf8·c语言gb2312·c语言gbk·c语言gb18030·gb2312转utf8·gbk转utf8
2401_890443021 天前
Linux 基础IO
linux·c语言
egoist20231 天前
【Linux仓库】超越命令行用户:手写C语言Shell解释器,解密Bash背后的进程创建(附源码)
linux·c语言·bash·xshell·环境变量·命令行参数·内建命令
superman超哥1 天前
仓颉语言中字典的增删改查:深度剖析与工程实践
c语言·开发语言·c++·python·仓颉
疑惑的杰瑞1 天前
【C】常见概念
c语言·编译原理
yyy(十一月限定版)1 天前
C语言——排序算法
c语言·开发语言·排序算法
黎雁·泠崖2 天前
指针收官篇:sizeof/strlen + 指针运算笔试考点全梳理
c语言·开发语言
lingran__2 天前
数据在内存中的存储详解(C语言拓展版)
c语言·开发语言