0

0

模板策略模式怎么实现 编译期多态方案

P粉602998670

P粉602998670

发布时间:2025-08-20 11:00:02

|

329人浏览过

|

来源于php中文网

原创

模板策略模式通过编译期多态在编译时确定策略,避免虚函数调用开销,提升性能;使用类型擦除可减少代码膨胀,而运行时动态切换策略可通过函数指针或跳转表实现,在灵活性与性能间取得平衡。

模板策略模式怎么实现 编译期多态方案

模板策略模式,本质上是为了解决算法族中,核心流程固定但具体步骤可变的问题。编译期多态方案,则是在编译时确定具体使用哪个策略,避免运行时的虚函数调用开销,提高性能。

解决方案

模板策略模式的核心在于定义一个模板方法,这个方法定义了算法的骨架,而将一些步骤延迟到子类去实现。编译期多态,意味着我们希望在编译时就确定使用哪个子类(策略)。这通常可以通过模板元编程来实现。

#include 
#include 

// 策略接口
template 
struct Strategy {
    virtual ~Strategy() = default;
    virtual void execute(T& data) = 0;
};

// 具体策略 A
template 
struct ConcreteStrategyA : public Strategy {
    void execute(T& data) override {
        std::cout << "ConcreteStrategyA executing with data: " << data << std::endl;
        data += 1;
    }
};

// 具体策略 B
template 
struct ConcreteStrategyB : public Strategy {
    void execute(T& data) override {
        std::cout << "ConcreteStrategyB executing with data: " << data << std::endl;
        data *= 2;
    }
};

// 模板类,接受策略类型作为模板参数
template  typename StrategyType>
class Context {
public:
    Context() : strategy(new StrategyType()) {} // 编译期确定策略类型
    ~Context() { delete strategy; }

    void processData(T& data) {
        strategy->execute(data);
    }

private:
    Strategy* strategy; // 指向策略对象的指针
};

int main() {
    int data = 5;
    Context contextA; // 编译期绑定 ConcreteStrategyA
    contextA.processData(data); // 输出 "ConcreteStrategyA executing with data: 5", data变为6

    int data2 = 10;
    Context contextB; // 编译期绑定 ConcreteStrategyB
    contextB.processData(data2); // 输出 "ConcreteStrategyB executing with data: 10", data2变为20
    return 0;
}

这段代码展示了基本的实现。

Context
类接受一个策略类型作为模板参数,并在构造函数中实例化该策略。这样,策略的选择就在编译时完成了。

如何避免代码膨胀?

编译期多态的一个潜在问题是代码膨胀,因为每个不同的策略组合都会生成一个新的

Context
类。 解决办法之一是使用类型擦除(Type Erasure)。 类型擦除允许你使用一个通用的接口来处理不同类型的对象,而不需要在编译时知道这些对象的具体类型。

#include 
#include 

template 
class AnyStrategy {
public:
    using ExecuteFunc = std::function;

    AnyStrategy(ExecuteFunc func) : executeFunc(func) {}

    void execute(T& data) {
        executeFunc(data);
    }

private:
    ExecuteFunc executeFunc;
};

template 
struct ConcreteStrategyA {
    void operator()(T& data) {
        std::cout << "ConcreteStrategyA executing with data: " << data << std::endl;
        data += 1;
    }
};

template 
struct ConcreteStrategyB {
    void operator()(T& data) {
        std::cout << "ConcreteStrategyB executing with data: " << data << std::endl;
        data *= 2;
    }
};

template 
class Context {
public:
    Context(AnyStrategy strategy) : strategy(strategy) {}

    void processData(T& data) {
        strategy.execute(data);
    }

private:
    AnyStrategy strategy;
};

int main() {
    int data = 5;
    Context contextA(AnyStrategy(ConcreteStrategyA()));
    contextA.processData(data);

    int data2 = 10;
    Context contextB(AnyStrategy(ConcreteStrategyB()));
    contextB.processData(data2);
    return 0;
}

这里

AnyStrategy
使用
std::function
来存储策略的执行函数,从而实现了类型擦除。
Context
类现在只需要一个模板参数
T
,而策略的选择在构造
AnyStrategy
对象时完成。

唱鸭
唱鸭

音乐创作全流程的AI自动作曲工具,集 AI 辅助作词、AI 自动作曲、编曲、混音于一体

下载

编译期策略选择的更高级应用场景有哪些?

除了基本的算法选择,编译期策略选择还可以用于更高级的场景,例如:

  • 性能优化: 根据编译时已知的信息(例如 CPU 特性),选择不同的优化策略。例如,如果编译器检测到支持 AVX 指令集,则选择使用 AVX 指令集的优化版本;否则,选择使用 SSE 指令集的版本。
  • 代码生成: 根据编译时配置,生成不同的代码。例如,可以根据编译时定义的宏,选择是否包含调试代码,或者选择使用不同的数据结构实现。
  • 安全策略: 根据编译时的安全配置,选择不同的安全策略。例如,可以根据编译时定义的权限级别,选择是否启用某些安全检查。

这些应用场景都需要更复杂的模板元编程技巧,例如使用

std::enable_if
来根据编译时条件选择不同的代码路径。

如何在运行时动态切换策略,同时尽量减少性能损失?

虽然我们讨论的是编译期多态,但有时候需要在运行时动态切换策略。一个折衷方案是使用函数指针或者

std::function

#include 
#include 

template 
void strategyA(T& data) {
    std::cout << "StrategyA executing with data: " << data << std::endl;
    data += 1;
}

template 
void strategyB(T& data) {
    std::cout << "StrategyB executing with data: " << data << std::endl;
    data *= 2;
}

template 
class Context {
public:
    using StrategyFunc = std::function;

    Context(StrategyFunc strategy) : strategy(strategy) {}

    void processData(T& data) {
        strategy(data);
    }

    void setStrategy(StrategyFunc newStrategy) {
        strategy = newStrategy;
    }

private:
    StrategyFunc strategy;
};

int main() {
    int data = 5;
    Context context(strategyA); // 初始策略为 strategyA
    context.processData(data); // 输出 "StrategyA executing with data: 5", data变为6

    context.setStrategy(strategyB); // 动态切换到 strategyB
    context.processData(data); // 输出 "StrategyB executing with data: 6", data变为12

    return 0;
}

虽然这引入了函数调用的开销,但避免了虚函数调用的开销。对于一些对性能要求不是特别高的场景,这是一个可以接受的方案。 另一种方式是使用跳转表(Jump Table),预先将所有可能的策略地址存储在一个数组中,然后根据索引来选择执行哪个策略。这可以减少动态查找策略的开销,但需要提前知道所有可能的策略。

相关专题

更多
java多态详细介绍
java多态详细介绍

本专题整合了java多态相关内容,阅读专题下面的文章了解更多详细内容。

14

2025.11.27

java多态详细介绍
java多态详细介绍

本专题整合了java多态相关内容,阅读专题下面的文章了解更多详细内容。

14

2025.11.27

treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

529

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

6

2025.12.22

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

989

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

50

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

202

2025.12.29

function是什么
function是什么

function是函数的意思,是一段具有特定功能的可重复使用的代码块,是程序的基本组成单元之一,可以接受输入参数,执行特定的操作,并返回结果。本专题为大家提供function是什么的相关的文章、下载、课程内容,供大家免费下载体验。

470

2023.08.04

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 3.1万人学习

PHP面向对象基础课程(更新中)
PHP面向对象基础课程(更新中)

共12课时 | 0.6万人学习

Golang基础入门到精通(第二季)
Golang基础入门到精通(第二季)

共49课时 | 2.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号