基础9 CRTP 与 Expression Templates

目录

一、奇异递归模版(CRTP)

二、表达式模板

🍉 概要

🍇 奇异递归模板模式(CRTP)

动机与原理

🍓  表达式模板(Expression Templates)

动机与原理

🍈 示例代码

🍒 示例程序对应的CRTP和编译树


一、奇异递归模版(CRTP)

作用:编译期多态,在compile斌阿姨阶段确定调用子类接口实现的静态的能力

#include <cassert>
#include <iostream>
#include <vector>template <typename Derived>
struct Base
{void name() { (static_cast<Derived *>(this))->impl(); }
};
struct D1 : public Base<D1>
{void impl() { std::cout << "D1::impl" << std::endl; }
};
struct D2 : public Base<D2>
{void impl() { std::cout << "D2::impl" << std::endl; }
};// 主函数入口
int main()
{Base<D1> derived1;Base<D2> derived2;D1 derived3;D2 derived4;func(derived1);func(derived2);func(derived3);func(derived4);return 0;
}

二、表达式模板

🍉 概要

最近在学习Eigen 线性代数计算库的代码,对其中很多C++的模板定义不甚理解,作为一名资深码农甚是惭愧。对于解决一个向量加法表达式v3 = v0 + v1 + v2,传统的编程方式存在中间内存的申请与释放、多次循环遍历等空间和时间的低效问题,不符合Eigen的高性能计算需求,为了提升性能,Eigen用了很多模板元编程技术,本文总结了其中两个知识点

  • 奇异递归模板模式(CRTP)
  • 表达式模板(Expression Templates)

🍇 奇异递归模板模式(CRTP)

动机与原理

  • 用父类提供统一的API 接口
  • 在Compile编译阶段确定调用子类接口实现的静态多态能力(相对于运行线虚函数的动态多态提升了运行时性能)
  • 实现原理如下图所示

🍓  表达式模板(Expression Templates)

动机与原理

  • 延迟计算表达式,从而可以将表达式传递给函数参数,而不是只能传表达式的计算结果
  • 节省表达式中间计算结果的临时存储空间,减少向量等线性代数计算的循环次数,从而减少整体计算的空间和时间成本
  • 实现原理如下图所示

🍈 示例代码

上面的原理图看起来还是比较难以理解的,下面以求解一个向量加法赋值表达式:v3 = v0 + v1 + v2 的程序代码为例解析原理图中的涵义

#include <cassert>
#include <iostream>
#include <vector>//CRTP中的基类模板
template <typename E>
class VecExpression {
public://通过将自己static_cast成为子类,调用子类的对应函数实现实现静态多态double operator[](size_t i) const { return static_cast<E const&>(*this)[i];     }size_t size()               const { return static_cast<E const&>(*this).size(); }};
//将自己作为基类模板参数的子类 - 对应表达式编译树中的叶节点
class Vec : public VecExpression<Vec> {std::vector<double> elems;public:double operator[](size_t i) const { return elems[i]; }double &operator[](size_t i)      { return elems[i]; }size_t size() const               { return elems.size(); }Vec(size_t n) : elems(n) {}Vec(std::initializer_list<double>init){for(auto i:init)elems.push_back(i);}//赋值构造函数可以接受任意父类VecExpression的实例,并且进行表达式的展开(对应表达式编译树中的赋值运算符节点)template <typename E>Vec(VecExpression<E> const& vec) : elems(vec.size()) {for (size_t i = 0; i != vec.size(); ++i) {elems[i] = vec[i];}}
};
//将自己作为基类模板参数的子类 - 对应表达式编译树中的二元运算符输出的内部节点
//该结构的巧妙之处在于模板参数E1 E2可以是VecSum,从而形成VecSum<VecSum<VecSum ... > > >的嵌套结构,体现了表达式模板的精髓:将表达式计算改造成为了构造嵌套结构
template <typename E1, typename E2>
class VecSum : public VecExpression<VecSum<E1, E2> > {E1 const& _u;E2 const& _v;
public:VecSum(E1 const& u, E2 const& v) : _u(u), _v(v) {assert(u.size() == v.size());}double operator[](size_t i) const { return _u[i] + _v[i]; }size_t size()               const { return _v.size(); }
};
//对应编译树上的二元运算符,将加法表达式构造为VecSum<VecSum... > >的嵌套结构
template <typename E1, typename E2>
VecSum<E1,E2> const operator+(E1 const& u, E2 const& v) {return VecSum<E1, E2>(u, v);
}
//主函数入口
int main() {//创建3个叶子节点Vec v0 = {1.0, 1.0, 1.0};Vec v1 = {2.0, 2.0, 2.0};Vec v2 = {3.0, 3.0, 3.0};//构建表达式 v0 + v1 + v2,赋值给v3,编译阶段形成表达式树Vec v3 = v0 + v1 + v2;//输出结算结果for (int i = 0; i < v3.size(); i ++) {std::cout <<" "<< v3[i];}std::cout << std::endl;
}
编译程序:
g++ -I /usr/local/include/eigen3/ -Wall -std=c++11 -fpermissive -g vecAdd.cpp -o vecAdd运行程序
./vecAdd程序输出:6 6 6

🍒 示例程序对应的CRTP和编译树

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/64916.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何在K8S集群中查看和操作Pod内的文件?

文章目录 一、理解Kubernetes中的Pod二、查看Pod内的文件三、操作Pod内的文件四、高级技巧五、常见问题与解决方案 在Kubernetes&#xff08;K8s&#xff09;集群中&#xff0c;Pod是最小的可部署单元&#xff1b; 一、理解Kubernetes中的Pod 在Kubernetes中&#xff0c;Pod是…

Zed调试宏 C语言错误日志 异常错误调试信息

1、C中的错误码 在C语言中通过返回错误码或设置全局的errno值来反馈错误问题。errno.h是一个头文件&#xff0c;它定义了一个全局变量errno&#xff0c;用于在程序中记录和报告错误的原因。这个机制主要用于处理系统调用或标准库函数出错时的错误反馈。当系统调用或库函数…

spring mvc | servlet :serviceImpl无法自动装配 UserMapper

纯注解SSM整合 解决办法&#xff1a; 在MybatisConfig添加 Configuration MapperScan("mapper")

分布式协同 - 分布式事务_TCC解决方案

文章目录 导图Pre流程图2PC VS 3PC VS TCC2PC&#xff08;Two-Phase Commit&#xff0c;二阶段提交&#xff09;3PC&#xff08;Three-Phase Commit&#xff0c;三阶段提交&#xff09;TCC&#xff08;Try-Confirm-Cancel&#xff09;2PC、3PC与TCC的区别2PC、3PC与TCC的联系 导…

源码分析之Openlayers中MousePosition鼠标位置控件

概述 本文主要介绍 Openlayers 中的MousePosition鼠标位置控件&#xff0c;该控件会创建一个元素在页面的右上方用来实时显示鼠标光标的位置坐标。该控件在实际应用很有效&#xff0c;可以实时获取鼠标位置&#xff0c;但是一般控件元素都会自定义。 源码分析 MousePosition…

脑肿瘤检测数据集,对9900张原始图片进行YOLO,COCO,VOC格式的标注

脑肿瘤检测数据集&#xff0c;对9900张原始图片进行YOLO&#xff0c;COCO&#xff0c;VOC格式的标注 数据集分割 训练组 70&#xff05; 6930图片 有效集 20&#xff05; 1980图片 测试集 10&#xff05; 990图片 预处理 静态裁剪&#xff1a; 24-82&…

SQL进阶技巧:如何计算商品需求与到货队列表进出计划?

目录 0 需求描述 1 数据准备 2 问题分析 3 小结 累计到货数量计算 出货数量计算

步进电机接线和stm32引脚分配

实验设备 24v&#xff08;12-48 v&#xff09;直流电源 stm32f103最小系统板 步进电机驱动器 采用混合式二相步进电机J-5718HBS2401-野火42步进电机&#xff0c;驱动器为野火EBF-MSD4805 本人参考接线方式如下&#xff1a; 如上图所示通常采用共阴接线方式&#xff0c;具体…

极乐 15.2.6 | 清爽版简约美观音乐软件,支持网易云歌单导入

极乐是一款使用起来非常轻松的音乐播放软件&#xff0c;它拥有清新简洁的画面&#xff0c;专注于音乐播放功能。最新版本全面升级了64位架构&#xff0c;带来了前所未有的性能提升和更稳定的体验。通过优化内存管理&#xff0c;降低了应用对系统资源的占用&#xff0c;确保设备…

Ubuntu Netlink 套接字使用介绍

Netlink 套接字 是 Linux 特有的一种 IPC&#xff08;进程间通信&#xff09;机制&#xff0c;用于用户态进程和内核模块之间的通信。它可以用来完成路由管理、设备通知、网络状态更新等任务。 1. Netlink 的基本工作原理 Netlink 是一种双向通信机制。Netlink 消息分为请求和…

4、mysql高阶语句

mysql高阶语句是对复杂的条件进行查询的操作。 排序—order by 加了desc表示由大到小 1、查询name和score&#xff0c;地址都是云南西路的按id进行由小到大排序 2、查询name和score&#xff0c;先按hobbid进行排序&#xff0c;再把结果按id进行排序 第一段字段必须要有相同的…

Docker部署GitLab服务器

一、GitLab介绍 1.1 GitLab简介 GitLab 是一款基于 Git 的开源代码托管平台&#xff0c;集成了版本控制、代码审查、问题跟踪、持续集成与持续交付&#xff08;CI/CD&#xff09;等多种功能&#xff0c;旨在为团队提供一站式的项目管理解决方案。借助 GitLab&#xff0c;开发…

hadoop中hive本地模式安装mysql源不成功

目录 1.更改DNS配置 2.替换yun源 3.替换掉后&#xff0c;在执行 4.重新安装mysql源 hive本地模式安装mysql源出错 yum install mysql mysql-server mysql-devel -y 解决&#xff1a; 1.更改DNS配置 vi /etc/resolv.conf 添加下面内容&#xff1a; nameserver 8.8.8.8 …

深入解析 Apache APISIX

以下是“第一部分&#xff1a;背景与概述”的示例写作内容&#xff0c;供你参考和使用。你可根据实际需求和篇幅进行增删或细化。 一、背景与概述 1. 高性能动态网关的意义 1.1 微服务架构下的网关角色与价值 随着微服务架构在企业级应用中日益普及&#xff0c;系统被拆分为…

RISC-V架构的压缩指令集介绍

1、压缩指令集介绍 RISC-V的压缩指令集&#xff08;C扩展&#xff09;‌是一种设计用于减少代码大小和提高性能的技术。标准的RISC-V指令是32位&#xff0c;压缩指令集可以将部分32位的指令用16位的指令替代&#xff0c;从未减小程序占用存储空间的大小&#xff0c;提高指令密…

Day13 苍穹外卖项目 工作台功能实现、Apache POI、导出数据到Excel表格

目录 1.工作台 1.1 需求分析和设计 1.1.1 产品原型 1.1.2 接口设计 1.2 代码导入 1.2.1 Controller层 1.2.2 Service层接口 1.2.3 Service层实现类 1.2.4 Mapper层 1.3 功能测试 1.4 代码提交 2.Apache POI 2.1 介绍 2.2 入门案例 2.2.1 将数据写入Excel文件 2.2.2 读取Excel文…

齐次矩阵包含平移和旋转

第一个矩阵旋转矩阵 A [ R 1 0 0 1 ] A\left[\begin{matrix}R_{1} & 0\\0 & 1\end{matrix}\right] A[R1​0​01​] 第一个平移矩阵 B [ 1 T 1 0 1 ] B\left[\begin{matrix}1 & T_{1}\\0 & 1\end{matrix}\right] B[10​T1​1​] C [ R 2 0 0 1 ] C\left[\be…

CSP-X2024山东小学组T4:刷题

题目链接 CSP-X2024山东小学组T4&#xff1a;刷题 题目描述 比赛之路多艰&#xff0c;做题方得提升。努力刷题的人在比赛中往往能取得很好的成绩&#xff0c;小红就是这样的人。 为了继续提升自己的编程实力&#xff0c;小红整理了一份刷题题单&#xff0c;并选中了题单中的…

集星獭 | 高性能编排:为实时数据集成而生!

概要介绍 服务编排作为集星獭驱动业务流、数据流中不可或缺的重要环节&#xff0c;其基于分布式架构打造&#xff0c;提供了高可用、易扩展的可视化流程任务调度功能。 原服务编排的设计初衷是专注于任务调度&#xff0c;提供高性能任务调度&#xff0c;但是在实时调用方面的…

已有 containerd 的情况下部署二进制 docker 共存

文章目录 [toc]学习目的开始学习dockerd启动 containerd准备配置文件启动 containerd 启动 docker准备配置文件启动 docker 环境验证停止 docker 和 containerd 学习目的 使用容器的方式做一些部署的交付&#xff0c;相对方便很多&#xff0c;不需要担心别人的环境缺少需要的依…