news 2026/5/4 10:49:38

第 4 篇:策略模式 (Strategy) —— 算法的热插拔艺术

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
第 4 篇:策略模式 (Strategy) —— 算法的热插拔艺术

专栏导读:你是否遇到过这种崩溃瞬间:产品卖给 A 客户要用 Modbus 协议,卖给 B 客户要用私有协议,卖给 C 客户要加密传输。你的代码里是不是充斥着无数的#ifdef CUSTOMER_A或者if (mode == 1)? 策略模式教你用 C 语言实现“多态”,让算法像 USB 设备一样,插上就能用,拔掉就消失。


1. 场景还原 (The Pain)

假设你正在维护一款智能门锁。核心功能是“验证密码”。 起初,只需要简单的明文比对。后来,高配版加入了指纹(Hash 校验),海外版要求符合安全规范(CRC32 或 AES)。

菜鸟的写法:条件判断的地狱

typedef enum { AUTH_SIMPLE, AUTH_CRC32, AUTH_HASH } AuthMode;

bool Verify_Password(uint8_t* input, uint8_t* stored, int len, AuthMode mode) {
if (mode == AUTH_SIMPLE) {
// 明文比对
return (memcmp(input, stored, len) == 0);
}
else if (mode == AUTH_CRC32) {
// 还要去包含 CRC 的头文件,耦合严重
uint32_t crc_in = Cal_CRC32(input, len);
uint32_t crc_store = *(uint32_t*)stored;
return (crc_in == crc_store);
}
else if (mode == AUTH_HASH) {
// ... 又是一大坨代码 ...
}
return false;
}

架构师的审视

这种代码违反了设计模式的最高天条:开闭原则 (Open-Closed Principle)—— 对扩展开放,对修改关闭。

  1. 耦合过重Verify_Password函数必须认识所有的算法。如果你要加一个新的“视网膜扫描”,你得修改这个经过测试的稳定函数,万一改坏了,旧功能也挂了。

  2. 代码臃肿:低配版芯片 Flash 只有 32KB,根本装不下 AES 库,但因为写在一个文件里,链接器很难剥离未用到的代码(除非编译器优化极强)。


2. 模式图解 (The Concept)

策略模式的核心是将**“做什么” (Interface)** 和“怎么做” (Implementation)分离。

  • Context (环境):持有算法的指针,不关心具体是哪个算法。

  • Strategy (接口):定义一套标准的函数指针结构体(V-Table)。

  • Concrete Strategy (具体策略):实现具体的算法。


3. 代码实战 (The Code)

C 语言实现多态的秘密武器:结构体 + 函数指针

3.1 定义接口 (The Interface)

这是所有算法必须遵守的契约。

// IValidator.h
#include <stdint.h>
#include <stdbool.h>

// 前置声明,类似 C++ 的 this 指针
typedef struct Validator_t Validator;

// 策略接口定义 (虚函数表)
typedef struct {
// 初始化算法
void (*init)(Validator* self);
// 执行计算
uint32_t (*calculate)(Validator* self, const uint8_t* data, uint32_t len);
// 销毁/清理 (可选)
void (*deinit)(Validator* self);
} ValidatorOps;

// 基类结构体
struct Validator_t {
const ValidatorOps* ops; // 指向虚表的指针
// 可以添加公共属性,如超时时间等
uint32_t timeout;
};

3.2 具体策略实现 (Concrete Strategies)

我们把每个算法写在独立的.c文件里,互不干扰。

策略 A:简单校验 (Simple)

// Validator_Simple.c
#include "IValidator.h"

static uint32_t simple_cal(Validator* self, const uint8_t* data, uint32_t len) {
uint32_t sum = 0;
for(int i=0; i<len; i++) sum += data[i];
return sum; // 简单的累加和
}

// 定义具体的虚表
static const ValidatorOps s_simple_ops = {
.init = NULL, // 不需要初始化
.calculate = simple_cal,
};

// 构造函数
void Validator_Simple_Init(Validator* v) {
v->ops = &s_simple_ops; // 核心:绑定指针
}

策略 B:CRC32 校验 (CRC32)

// Validator_CRC32.c
#include "IValidator.h"
#include "stm32_crc.h" // 依赖硬件库

static void crc_init(Validator* self) {
__HAL_RCC_CRC_CLK_ENABLE(); // 开启硬件时钟
}

static uint32_t crc_cal(Validator* self, const uint8_t* data, uint32_t len) {
return HAL_CRC_Calculate(&hcrc, (uint32_t*)data, len);
}

static const ValidatorOps s_crc32_ops = {
.init = crc_init,
.calculate = crc_cal,
};

void Validator_CRC32_Init(Validator* v) {
v->ops = &s_crc32_ops;
}

3.3 业务调用 (Context)

// main.c
#include "IValidator.h"

// 全局的校验器实例
Validator g_current_validator;

// 根据配置选择策略 (Factory 模式的雏形)
void System_Init() {
uint8_t config = Read_Flash_Config();

if (config == MODE_HIGH_END) {
Validator_CRC32_Init(&g_current_validator);
} else {
Validator_Simple_Init(&g_current_validator);
}

// 初始化选定的算法
if (g_current_validator.ops->init) {
g_current_validator.ops->init(&g_current_validator);
}
}

// 业务逻辑:完全不需要知道底层是 CRC 还是 Sum
bool Verify_Packet(uint8_t* data, int len, uint32_t expected) {
// 多态调用!
uint32_t result = g_current_validator.ops->calculate(&g_current_validator, data, len);

return (result == expected);
}

4. 内存与性能分析 (The Cost)

空间开销

  • Flash: 这种写法对 Linker 非常友好。如果你在System_Init里没有调用Validator_CRC32_Init,聪明的链接器(开启--gc-sections)会自动把Validator_CRC32.c的代码完全剔除。这就实现了**“用多少,链多少”**。

  • RAM: 每个策略实例只需要一个指针大小(4 字节)。虚表ValidatorOps声明为const,存放在 Flash 中,不占 RAM。

时间开销

  • 间接调用:ops->calculate(...)是一次间接函数调用 (Indirect Call)

  • 汇编视角:

    • 直接调用:BL 0x08001234(跳转到固定地址)

    • 间接调用:LDR R0, [R1]; BLX R0(先读地址再跳转)

  • 性能损耗: 相比直接调用,多了 2-3 个指令周期。

  • 架构师建议:不要在百万次的极速循环内部使用策略模式(例如像素级渲染)。但对于通信包校验、按键处理、传感器读取这种毫秒级任务,性能损耗可以忽略不计,换来的是巨大的架构灵活性。


5. 变种与延伸 (The Evolution)

5.1 运行时动态切换 (Hot Swap)

这就是标题中“热插拔”的由来。 比如一个无线接收模块,信号好时用Strategy_Fast(无校验,速度快),信号差时自动切换为Strategy_Reliable(高强度校验)。 你只需要执行v->ops = &s_reliable_ops;,业务层代码无需任何变动。

5.2 依赖注入与打桩 (Mocking for Test)

这是策略模式在单元测试中的杀手级应用。 在 PC 上跑单元测试时,无法调用HAL_CRC_Calculate(因为没有硬件)。 你可以写一个Validator_Mock.c,在里面伪造数据。

// Test_Main.c
void Test_Logic() {
Validator v;
Validator_Mock_Init(&v); // 注入假对象

// 测试业务逻辑,此时 calculate 调用的是假的桩函数
Verify_Packet(..., ...);
}

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/2 4:16:30

Sentinel 流控配置案例:两次请求的时间间隔必须在3秒以上

要实现“两次请求的时间间隔必须在3秒以上”这种需求&#xff0c;需要根据具体的业务场景选择合适的Sentinel配置方式。本文博主将介绍几种实现方案&#xff1a; 方案1&#xff1a;使用排队等待流控效果&#xff08;最接近需求&#xff09; # Sentinel规则配置&#xff08;对…

作者头像 李华
网站建设 2026/5/2 3:35:50

告别低效内耗!这款CRM营销管理系统,让职场人少走80%的弯路

你是否每天被这些职场难题裹挟&#xff0c;身心俱疲却业绩平平&#xff1f; ✅ 营销获客像“大海捞针”&#xff1a;砸钱投渠道、熬夜做推广&#xff0c;线索来了却杂乱无章&#xff0c;高价值客户藏在信息堆里无人问津&#xff0c;获客成本居高不下&#xff0c;转化效果惨不忍…

作者头像 李华
网站建设 2026/4/17 17:01:56

AI智能生成微信红包封面工具 | 一键制作个性化红包封面系统源码

AI微信红包封面生成器源码是一款开源的微信红包封面生成工具&#xff0c;由前腾讯微信后台开发工程师「idoubi」开发并开源。项目名为“AI Cover”&#xff0c;旨在利用人工智能技术为用户提供个性化的微信红包封面生成服务。支持用户登录、付费方案和积分系统&#xff0c;提供…

作者头像 李华
网站建设 2026/4/21 21:10:36

5 款 AI 写论文哪个好?实测后发现:虎贲等考 AI 凭这 3 点碾压同类

在学术写作数字化转型的浪潮中&#xff0c;无数科研人、学子被 “文献难寻、数据失真、格式混乱、查重焦虑” 等痛点困扰。虎贲等考 AI 智能写作平台&#xff08;官网&#xff1a;https://www.aihbdk.com/&#xff09;应势而生&#xff0c;作为一款深度融合前沿人工智能技术的专…

作者头像 李华
网站建设 2026/4/27 16:12:21

利用Snyk发现与修复漏洞:守护软件安全,保障业务稳定

利用Snyk发现与修复漏洞&#xff1a;守护软件安全&#xff0c;保障业务稳定 作为一名开发者&#xff0c;代码的安全性和稳定性对我们保持理智至关重要。当代码不稳定或容易出错时&#xff0c;修复起来会令人沮丧且耗时&#xff0c;导致不知所措和心力交瘁的感觉。 这就是Snyk的…

作者头像 李华