条款44:将与参数无关的代码抽离templates

这篇具有很好参考价值的文章主要介绍了条款44:将与参数无关的代码抽离templates。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

1.前言

Template是个节省时间和避免代码重复的一个方法。不再需要键入20个类似的classes而每一个带有15个成员函数,只需要键入一个class template,留给编译器去具现化那20个你需要的相关classes和300个函数。(class template的成员函数只有在被使用时才被暗中具现化,只有在这300个函数的每一个都被使用,你才会获得这300个函数),function template有类似的诉求。替换写许多函数,只需写一个function template,然后让编译器做剩余的事情。

但如果不小心,使用template可能会导致代码膨胀:其二进制码带着重复的代码,数据。其结果可能是源码看起来合身且整齐,但目标码(object code)却不是那么一回事。

2.实例分析

当你编写某个函数,而你明白其中某些部分的实现码和另外一个函数的实现码实质相同,我们一般会抽象出两个函数的共同部分,把它放进第三个函数中,然后令原先两个函数调用这个新函数。同样的道理,如果你正在编写某个class,而其中的某些部分和另外一个class的某些部分相同,你也不会重复这共同的一部分,取而代之的是你会把共同部分搬移到新的class去,然后使用继承或复合,令原先的1classes取用这共同特性。而原的classes的互异部分仍然留在原位置不动。

编写templates时,也是做相同的分析,以相同的方式避免重复,但其中有个窍门。在non-template代码中,重复十分明确:你可以看到两个函数或两个classes之间有所重复。然而在template代码中,重复是隐晦的:毕竟只有一份template源码,所以必须训练自己去感受当template被具现化时可能发生的重复。

举个例子,假设你想为固定尺寸的正方矩阵编写一个temolate。该矩阵的形式之一是支持逆矩阵运算(matrix inversion)。

template<typename T,std::size_t n>//template支持n,x,n矩阵,
                                    //元素是类型T的objects;见以下关于//size_t参数信息
class SquareMatrix{    
    public:
        ...
        void invert();//求逆矩阵

};           

这个template接受一个类型参数T,除此以外还接受一个类型为size_t的参数那是个非类型参数(non-type parameter)。这种参数和类型参数比较起来不常见,但它们完全合法,就比如本例:

SquareMatrix<double,5> sml;
...
sml.invert();//调用SquareMatrix<double,5>::invert
SquareMatrix<double,10> sm2;
...
sm2.invert();//调用SquareMatrix<double,10>::invert

这会具象化两份invert,这些函数并非完完全全相同,因为其中一个操作的是5*5矩阵而另一个操作的是10*10矩阵,但除来常量5和10,两个函数的其它部分完全相同。这是template引出代码膨胀的一个典型例子。

如果你能看到两个函数完全相同,除了使用5和使用10的区别,本能是为它们建立一个带数值参数的函数,然后以5和10来调用这个带参数的函数,而不重复代码。下面是对SquareMatrix的第一次修改:

template<typename T>
class SquareMatrixBase{

    protected:
        ...
        void invert(std::size_t matrixSize);//以给定的尺寸求逆矩阵
        ...

};
tempalate<typename T,std::size_t n>
class SquareMatrix:private SquareMatrixBase<T>
{
    private:
        using SquareMatrixBase<T>::invert;//避免遮掩base 版本的invert

    public:
        ...
        void invert() 
        {
            this->invert(n);
        }
};

正如所看到的,带参数的invert位于base class SquareMatrixBase中,和SquareMatrix一样,SquareMatrixBase也是个template,不同的是它只对”矩阵元素对象的类型“参数化,不对矩阵的尺寸参数化。因此对于给定之元素对象类型,所有矩阵共享同一个SquareMatrixBase class。它们也将因此共享这唯一一个class内的invert。

SquareMatrixBase::invert只是企图成为“避免derived class代码重复”的一种方法,所以它以protected替换public。调用它造成的额外成本应该是0,因为derived classes的inverts调用base class版本时用的是Inline调用。这些函数使用“this->”记号,假设不这样做,模板化基类内的函数名称会被derived classes掩盖。也请注意SquareMatrix和SquareMatrixBase之间的继承关系是private。这反映一个事实:这里的base class只是为了帮助derived class实现,不是为了表现SquareMatrix和SquareMatrixBase之间的is-a关系。

当然,这里目前还存在一些问题:SquareMatrixBase::invert如何知道该数据是什么数据,虽然可以从参数中知道矩阵尺寸,但如何知道哪个特定的矩阵的数据在哪?想必只有derived class知道,Derived class如何联络其base class做逆运算操作动作。一个可能的做法是为了SquareMatrixBase::invert添加另一个参数,也许是个指针,指向一块用来放置矩阵数据的内存起点,这个方法行得通。但invert不是唯一一个可写为“形式与尺寸无关并可移动到SquareMatrixBase内的SquareMatrixBase内“的SquareMatrix函数。如果有若干这样的函数,我们唯一要做的就是找出保存矩阵元素值得那块内存。我们可以对所有这样得函数添加一个额外得参数,却得一次又一次地告诉SquareMatrixBase相同地信息,这样似乎不好。

另一个办法是令SquareMatrixBase储存一个地址,指向矩阵数值所在得内存。而它只要存了那些东西,就可能存储矩阵得大小。类似这样:

template<typename T>
class SquareMatrixBase{

    protected:
        SquareMatrixBase(std::size_t,T* pMem):size(n),pData(pMem)//存矩阵大
                                                                //小和一个指针,指向矩阵数值
        {

            
        }

        void setDataPtr(T* ptr)
        {
            pData=ptr;
        }
        ....
        private:
        std::size_t size;//矩阵大小
        T* pData;//指针,指向矩阵内容
};

这允许derived classes决定内存分配方式。某些实现版本也许会决定将矩阵数据存储在SquareMatrix对象内部:

template<typename T,std::size_t n>
class SquareMatrix:private SquareMatrixBase<T>
{

    public:
        SquareMatrix():SquareMatrixBase<T>(n,data) { }//送出矩阵大小和数据指针给base class
    ....
    private:
        T data[n*n];

};

这种类型的对象不需要动态分配内存,但对象自身可能性非常大,另一种做法是把每一个矩阵的数据放进heap(也就是通过new 来分配内存);

template<typename T,std::size_t n>
class SquareMatrix:private SquareMatrixBase<T>
{
    public:
        SquareMatrix():SquareMatrixBase<T>(n,0),pData(new T[n*n])//将base 的数据指针设为null
        {                                                    //为矩阵分配内存
            this->setDataPtr(pData.get());//将它的一个副本交给base class
        }
   
    private:
        boost::scoped_array<T> pData;
}

3.总结

(1)Template生成多个classes和多个函数,所以任何template代码都不该与某个造成膨胀的template参数产生联系;

(2)因非类型模板参数而造成的代码膨胀,往往可以消除,做法是以函数参数或class成员变量替换template参数

(3)因类型参数(type parameters)而造成的代码膨胀,往往可降低,做法是让带有完全相同二进制表述的具现类型共享实现码。文章来源地址https://www.toymoban.com/news/detail-802464.html

到了这里,关于条款44:将与参数无关的代码抽离templates的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • .net8系列-04图文并茂手把手教你配置Swagger支持token以及实现Swagger扩展,Swagger代码单独抽离

    接上篇文章,我们当前已完成如下内容: 创建应用成功 创建接口成功 配置Swagger实现接口注释和版本控制 本文章主要内容为: 配置Swagger支持token传值测试接口 添加如下代码 文件目录:xiaojinWebApplicationxiaojinWebApplicationProgram.cs 打开接口测试页面 配置Token 观察页面 我们发现

    2024年04月27日
    浏览(27)
  • 代码随想录day44

    完全背包 其实就是每个物品可以使用无数次,给我们一个容器,装满这个容器的最大价值是多少。 思路: 如果求组合数就是外层for循环遍历物品,内层for遍历背包。 如果求排列数就是外层for遍历背包,内层for循环遍历物品。 完全背包的组合和排序 518. 零钱兑换 II 题目 给你

    2023年04月17日
    浏览(40)
  • 代码随想录 day44 完全背包

    class Solution { public:     int change(int amount, vectorint coins) {         vector int dp(amount+1,0);         dp[0]=1;         for(int i=0;icoins.size();i++){             for(int j=coins[i];j=amount;j++){                 dp[j]+=dp[j-coins[i]];             }         }  

    2024年02月15日
    浏览(33)
  • day44代码训练|动态规划part06

    完全背包和01背包问题唯一不同的地方就是,每种物品有无限件 。 1. dp数组的含义 dp[i][j] 0-i物品,重量为j的容量时,最大的价值 2. 递推公式 dp[i][j] = max(dp[i-1][j],dp[i][j-weight[i]]+value[i]); 两种状态,不用物品i的话,直接是用dp[i-1][j] 选用物品的话,为了重复使用物品i,其实是

    2024年02月03日
    浏览(29)
  • 代码随想录第44天|动态规划:完全背包理论基础 518.零钱兑换II 377. 组合总和 Ⅳ

    代码随想录 (programmercarl.com) 动态规划之完全背包,装满背包有多少种方法?组合与排列有讲究!| LeetCode:518.零钱兑换II_哔哩哔哩_bilibili 完全背包和01背包问题唯一不同的地方就是,每种物品有无限件 。 完全背包中的物品可以添加多次,所以要从小到大遍历: 518. 零钱兑换

    2024年04月25日
    浏览(35)
  • vue中相同逻辑如何进行抽离

    可以使用 vue 里面的混入( mixin )技术。混入( mixin )提供了一种非常灵活的方式,来将 vue 中相同的业务逻辑进行抽离。 例如: 在 data 中有很多是公用数据 引用封装好的组件也都是一样的 methods、watch、computed 中也都有大量的重复代码 当然这个时候可以将所有的代码重复

    2024年02月14日
    浏览(28)
  • 【群智能算法改进】基于二次插值策略的改进白鲸优化算法 改进后的EBWO[3]算法【Matlab代码#44】

    白鲸优化算法 (BWO,beluga whale optimization) 是2022 年在白鲸游泳、捕鲸及跌倒等行为中得到启发而提出的一种新型基于种群的元启发式算法。BWO 主要对白鲸游泳、捕食及跌倒 (坠落) 等行为进行模拟,其对应探索、开发及鲸鱼坠落三个阶段。BWO 当中鲸落概率与平衡因子均为自适应

    2024年02月11日
    浏览(35)
  • maven打包抽离第三方jar

    gitee地址:ruoyi-pom-config: 用来修改ruoyi-cloud的pom文件示例,第三方jar包抽取打包 以若依微服务项目为说明示例,需要用git下载下来结合理解,后端项目结构 图一:  红圈一的pom为最父级的pom文件 最父级 pom文件内容如下  先重点看最父级pom文件,下面图中的红圈部分 图二:

    2024年02月20日
    浏览(45)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包