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++ 中推荐用智能指针管理策略对象,结合开闭原则,新增算法只需扩展策略类,无需修改原有代码。
相关推荐
郝学胜-神的一滴1 分钟前
[简化版 GAMES 101] 计算机图形学 03:线性代数下
开发语言·c++·线性代数·图形渲染
Dxy12393102162 分钟前
如何使用 ECharts 绘制 K 线图
开发语言·javascript
fe7tQnVan2 分钟前
从玩具到生产:基于 ChromaDB 打造工程级 RAG 系统
开发语言·c#
一路向北he9 分钟前
esp32库依赖
c语言·c++·算法
Frank_refuel11 分钟前
QT->信号与槽详解下补充(概述、使用、自定义、连接方式、其他说明)
开发语言·qt
happymaker062612 分钟前
VueCli标准化工程中的组件通信操作
开发语言·前端·javascript
ySq0REx0115 分钟前
.NET 10 & C# 14 New Features 新增功能介绍-.NET CLI工具改进
开发语言·c#·.net
Howrun77719 分钟前
C++ 项目测试全指南:从 0 基础到落地实操
开发语言·c++·log4j
小灰灰搞电子19 分钟前
Qt UI 线程详解-阻塞与解决方案
开发语言·qt·ui
YYYing.20 分钟前
【Linux/C++网络篇(二) 】TCP并发服务器演进史:从多进程到Epoll的进化指南
linux·服务器·网络·c++·tcp/ip