C++ 策略模式实战:从原理到落地

策略模式是 C++ 中经典的行为型设计模式 ,核心思想是「将算法封装成独立策略类,让算法可动态替换」,既能避免大量if-else分支,又能提高代码扩展性。下面从「原理拆解 + 实战案例」两部分,带你彻底掌握策略模式的落地用法。


一、策略模式核心原理

1. 核心角色(3 个)

  • 抽象策略类(Strategy):定义算法的统一接口(纯虚函数),所有具体策略需实现该接口。
  • 具体策略类(ConcreteStrategy):实现抽象策略接口,封装不同的算法逻辑。
  • 上下文类(Context):持有策略对象的引用,对外提供统一调用入口,可动态切换策略。

2. 核心优势

  • 消除分支判断:替代if-else/switch,代码更简洁。
  • 开闭原则:新增算法只需加新策略类,无需修改原有代码。
  • 算法复用:策略类可在不同场景复用。

二、实战案例:电商订单折扣计算

场景说明

电商订单需根据用户类型(普通用户、VIP、超级 VIP)计算不同折扣,若用if-else会导致代码臃肿,用策略模式可优雅解决。

完整实战代码

cpp

运行

复制代码
#include <iostream>
#include <string>
#include <memory> // 智能指针,避免内存泄漏

// 1. 抽象策略类:折扣计算接口
class DiscountStrategy {
public:
    // 纯虚函数:计算折扣后价格
    virtual double calculate(double originalPrice) = 0;
    // 虚析构:确保子类析构被调用
    virtual ~DiscountStrategy() = default;
};

// 2. 具体策略类1:普通用户(无折扣)
class NormalUserDiscount : public DiscountStrategy {
public:
    double calculate(double originalPrice) override {
        std::cout << "普通用户:无折扣\n";
        return originalPrice;
    }
};

// 具体策略类2:VIP用户(9折)
class VipUserDiscount : public DiscountStrategy {
public:
    double calculate(double originalPrice) override {
        std::cout << "VIP用户:9折优惠\n";
        return originalPrice * 0.9;
    }
};

// 具体策略类3:超级VIP用户(8折+满100减10)
class SuperVipUserDiscount : public DiscountStrategy {
public:
    double calculate(double originalPrice) override {
        std::cout << "超级VIP用户:8折+满100减10\n";
        double price = originalPrice * 0.8;
        // 满100减10
        if (price >= 100) {
            price -= 10;
        }
        return price;
    }
};

// 3. 上下文类:订单(封装策略调用)
class Order {
private:
    // 持有策略对象(智能指针自动管理内存)
    std::unique_ptr<DiscountStrategy> discount_strategy_;
    double original_price_; // 原价
public:
    // 构造函数:初始化原价+策略
    Order(double price, std::unique_ptr<DiscountStrategy> strategy)
        : original_price_(price), discount_strategy_(std::move(strategy)) {}

    // 动态切换策略(核心:运行时替换算法)
    void setDiscountStrategy(std::unique_ptr<DiscountStrategy> strategy) {
        discount_strategy_ = std::move(strategy);
    }

    // 对外统一接口:计算最终价格
    double getFinalPrice() {
        if (!discount_strategy_) {
            throw std::runtime_error("未设置折扣策略!");
        }
        return discount_strategy_->calculate(original_price_);
    }
};

// 测试代码
int main() {
    // 订单原价200元
    double original_price = 200.0;

    // 1. 普通用户订单
    Order normal_order(original_price, std::make_unique<NormalUserDiscount>());
    std::cout << "普通用户最终价格:" << normal_order.getFinalPrice() << "元\n";

    // 2. 切换为VIP策略
    normal_order.setDiscountStrategy(std::make_unique<VipUserDiscount>());
    std::cout << "VIP用户最终价格:" << normal_order.getFinalPrice() << "元\n";

    // 3. 切换为超级VIP策略
    normal_order.setDiscountStrategy(std::make_unique<SuperVipUserDiscount>());
    std::cout << "超级VIP最终价格:" << normal_order.getFinalPrice() << "元\n";

    return 0;
}

代码运行结果

plaintext

复制代码
普通用户:无折扣
普通用户最终价格:200元
VIP用户:9折优惠
VIP用户最终价格:180元
超级VIP用户:8折+满100减10
超级VIP最终价格:150元

代码关键解析

  1. 抽象策略类DiscountStrategy :定义了calculate纯虚函数,是所有折扣算法的统一接口。
  2. 具体策略类NormalUserDiscount/VipUserDiscount/SuperVipUserDiscount分别实现不同折扣逻辑,彼此独立。
  3. 上下文类Order
    • 持有策略对象指针,对外隐藏具体算法细节;
    • 提供setDiscountStrategy方法,支持运行时动态切换策略
    • getFinalPrice是统一调用入口,用户无需关心具体折扣逻辑。
  4. 智能指针 :使用std::unique_ptr管理策略对象,避免手动释放内存,符合 C++ 现代编程规范。

三、策略模式实战扩展技巧

  1. 策略工厂:若策略类型多,可封装「策略工厂类」,根据用户类型自动创建对应策略对象,进一步简化调用。
  2. 结合 lambda:简单策略可直接用 lambda 表达式替代策略类,减少代码量(适合轻量级场景)。
  3. 策略配置化:将策略与配置文件绑定(如 JSON),无需改代码即可调整算法逻辑。

总结

  1. 策略模式核心是「算法与使用场景解耦」,通过抽象策略接口 + 具体策略类,实现算法的动态替换。
  2. 实战中需明确「抽象策略接口」「具体策略实现」「上下文封装」三个核心角色,避免过度设计。
  3. C++ 中推荐用智能指针管理策略对象,结合开闭原则,新增算法只需扩展策略类,无需修改原有代码。
相关推荐
CS创新实验室2 小时前
从硬件抽象到意图对齐:论 AI 时代操作系统演进的逻辑必然与 OpenClaw 的范式价值
人工智能·操作系统·策略模式
IT北辰2 小时前
不规则 Excel“数据提取——教师课表自动汇总实战
开发语言·爬虫·python
勿芮介2 小时前
【研发工具】OpenClaw基础环境安装全教程-Node\NVM\PNPM\Bash
开发语言·node.js·bash·ai编程
JamesYoung79712 小时前
第七部分 — 存储 数据建模与迁移提示
java·开发语言·数据结构
爱装代码的小瓶子2 小时前
【c++与Linux进阶】线程篇 -互斥锁
linux·c++·算法
一叶落4382 小时前
LeetCode 149. 直线上最多的点数(C语言详解 | 斜率 + 最大共线点)
数据结构·c++·算法·leetcode
仰泳的熊猫2 小时前
题目2267:蓝桥杯2016年第七届真题-取球博弈
数据结构·c++·算法·蓝桥杯·深度优先·图论
大尚来也2 小时前
超越“传参”:HTTP GET与POST的深度辨析与场景选型指南
开发语言
大鹏说大话2 小时前
破局与重构:微服务架构的演进之路、核心挑战与基石组件
开发语言