草庐IT

没有动态调度开销的基类抽象方法的 C++ 习惯用法?

coder 2024-02-15 原文

在 C++ 中,是否有任何方法可以在不将方法声明为 的情况下拥有“抽象”基类方法(,从基类声明和调用,但在子类中实现)虚拟?

当然,这个问题只适用于不需要多态性的情况(从未使用过的基类型的指针/引用)。请考虑以下事项:

#define NO_OPT asm volatile ("");  // to prevent some compiler optimization

template<typename DerivedType>
void doSomething(DerivedType& d) { d.foo(); }

namespace test1 {

    struct Base {
        inline void foo()
        {
            // ... do common stuff pre-call ...
            foo_impl();
            // ... do common stuff post-call ...
        }
        virtual void foo_impl() = 0; // the abstract method
    };
    struct D1 : public Base { virtual void foo_impl() final { NO_OPT } };
    struct D2 : public Base { virtual void foo_impl() final { NO_OPT } };

    // Then the usage of D1, D2, ..., DN, could be as follows:

    void go() {

        D1 d1; doSomething(d1);
        D2 d2; doSomething(d2);

        for ( long i=0; i < 5000000000; i++ ) {
            // this loop takes [9] seconds
            doSomething(d2);
        }
    }
}

请注意,在这种情况下不需要多态性,编译器有很多优化机会。

但是,我在最新的 g++ (4.8.2) 和 clang (3.4) 中对启用了 -O3 优化的代码进行了基准测试,包括链接时间 (LTO),并且它比以下替代实现(使用模板而不是虚拟方法)慢得多:

namespace test2 {

    template<typename DerivedType>
    struct Base : public DerivedType  // inheritance in reverse
    {
        inline void foo()
        {
            // ... do common stuff pre-call ...
            DerivedType::foo_impl();
            // ... do common stuff post-call ...
        }
    };
    struct D1 { void foo_impl() { NO_OPT } };
    struct D2 { void foo_impl() { NO_OPT } };

    void go() {
        Base<D1> d1; doSomething(d1);
        Base<D2> d2; doSomething(d2);

        for ( long i=0; i < 5000000000; i++ ) {
            // this loop takes [3] seconds
            doSomething(d2);
        }
    }
}

g++clang 非常一致,每个编译优化代码都需要 9 秒来执行 test1 循环,但只需要 3 秒执行 test2 循环。因此,即使 test1 逻辑不需要动态调度,因为所有调用都应该能够在编译时解决,但它仍然慢得多。

所以重申我的问题:当不需要多态性时,有没有一种方法可以使用方便的直接类继承(如 test1)来实现这种“抽象方法”行为),但没有虚函数的性能损失(如 test2 中所实现)?

最佳答案

为什么要显式调用运行时支持以应对您没有且不想被视为直截了当的情况?这也会让人类感到困惑。模板是这项工作的完美工具。

您需要一个通用的界面。鉴于您的示例模板实现,您甚至不需要拥有它的类型之间有任何明显的关系,只需要它们确实拥有它即可。

经典CRTP方式是

template<class d>
struct b { 
    void foo() { bark(); static_cast<d*>(this)->foo_impl(); bite(); }
};

struct d1 : b<d1> { void foo_impl() { fuzzbang("d1"); } };
struct d2 : b<d2> { void foo_impl() { fizzbuzz("d2"); } };

人们以前会看到的,这是符合要求的惯用语,但是您的

class m1 { protected: void foo_impl() { f1("1"); } };
class m2 { protected: void foo_impl() { f2("2"); } };

template<class m>
struct i : private m { void foo() { bark(); m::foo_impl(); bite(); } };

它也有很多值得推荐的地方:不需要强制转换,而且意图很明确。


我认为您无法合理地将优化器在具有嵌入式 volatile 访问的代码上的结果与在没有它们的情况下对类似代码的结果进行比较。在您的实际代码中尝试这两个实现。你正在做的任何事情的性能 50 亿次几乎肯定是由高速缓存效应而不是指令数决定的。

关于没有动态调度开销的基类抽象方法的 C++ 习惯用法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18244139/

有关没有动态调度开销的基类抽象方法的 C++ 习惯用法?的更多相关文章

  1. ruby - 难道Lua没有和Ruby的method_missing相媲美的东西吗? - 2

    我好像记得Lua有类似Ruby的method_missing的东西。还是我记错了? 最佳答案 表的metatable的__index和__newindex可以用于与Ruby的method_missing相同的效果。 关于ruby-难道Lua没有和Ruby的method_missing相媲美的东西吗?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/7732154/

  2. ruby-on-rails - rails 目前在重启后没有安装 - 2

    我有一个奇怪的问题:我在rvm上安装了ruby​​onrails。一切正常,我可以创建项目。但是在我输入“railsnew”时重新启动后,我有“程序'rails'当前未安装。”。SystemUbuntu12.04ruby-v"1.9.3p194"gemlistactionmailer(3.2.5)actionpack(3.2.5)activemodel(3.2.5)activerecord(3.2.5)activeresource(3.2.5)activesupport(3.2.5)arel(3.0.2)builder(3.0.0)bundler(1.1.4)coffee-rails(

  3. ruby-on-rails - 如何优雅地重启 thin + nginx? - 2

    我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server

  4. ruby - 在没有 sass 引擎的情况下使用 sass 颜色函数 - 2

    我想在一个没有Sass引擎的类中使用Sass颜色函数。我已经在项目中使用了sassgem,所以我认为搭载会像以下一样简单:classRectangleincludeSass::Script::FunctionsdefcolorSass::Script::Color.new([0x82,0x39,0x06])enddefrender#hamlengineexecutedwithcontextofself#sothatwithintemlateicouldcall#%stop{offset:'0%',stop:{color:lighten(color)}}endend更新:参见上面的#re

  5. 没有类的 Ruby 方法? - 2

    大家好!我想知道Ruby中未使用语法ClassName.method_name调用的方法是如何工作的。我头脑中的一些是puts、print、gets、chomp。可以在不使用点运算符的情况下调用这些方法。为什么是这样?他们来自哪里?我怎样才能看到这些方法的完整列表? 最佳答案 Kernel中的所有方法都可用于Object类的所有对象或从Object派生的任何类。您可以使用Kernel.instance_methods列出它们。 关于没有类的Ruby方法?,我们在StackOverflow

  6. ruby-on-rails - Rails 3,嵌套资源,没有路由匹配 [PUT] - 2

    我真的为这个而疯狂。我一直在搜索答案并尝试我找到的所有内容,包括相关问题和stackoverflow上的答案,但仍然无法正常工作。我正在使用嵌套资源,但无法使表单正常工作。我总是遇到错误,例如没有路线匹配[PUT]"/galleries/1/photos"表格在这里:/galleries/1/photos/1/edit路线.rbresources:galleriesdoresources:photosendresources:galleriesresources:photos照片Controller.rbdefnew@gallery=Gallery.find(params[:galle

  7. ruby-on-rails - 有没有办法为 CarrierWave/Fog 设置上传进度指示器? - 2

    我在Rails应用程序中使用CarrierWave/Fog将视频上传到AmazonS3。有没有办法判断上传的进度,让我可以显示上传进度如何? 最佳答案 CarrierWave和Fog本身没有这种功能;你需要一个前端uploader来显示进度。当我不得不解决这个问题时,我使用了jQueryfileupload因为我的堆栈中已经有jQuery。甚至还有apostonCarrierWaveintegration因此您只需按照那里的说明操作即可获得适用于您的应用的进度条。 关于ruby-on-r

  8. ruby - 没有类方法获取 Ruby 类名 - 2

    如何在Ruby中获取BasicObject实例的类名?例如,假设我有这个:classMyObjectSystem我怎样才能使这段代码成功?编辑:我发现Object的实例方法class被定义为returnrb_class_real(CLASS_OF(obj));。有什么方法可以从Ruby中使用它? 最佳答案 我花了一些时间研究irb并想出了这个:classBasicObjectdefclassklass=class这将为任何从BasicObject继承的对象提供一个#class您可以调用的方法。编辑评论中要求的进一步解释:假设你有对象

  9. ruby - 使用 `+=` 和 `send` 方法 - 2

    如何将send与+=一起使用?a=20;a.send"+=",10undefinedmethod`+='for20:Fixnuma=20;a+=10=>30 最佳答案 恐怕你不能。+=不是方法,而是语法糖。参见http://www.ruby-doc.org/docs/ProgrammingRuby/html/tut_expressions.html它说Incommonwithmanyotherlanguages,Rubyhasasyntacticshortcut:a=a+2maybewrittenasa+=2.你能做的最好的事情是:

  10. ruby - 没有轨道的 ActiveRecord 时区 - 2

    我在非Rails项目中使用ActiveRecord。在Rails中,我可以这样做:config.time_zone='EasternTime(US&Canada)'config.active_record.default_timezone='EasternTime(US&Canada)'但如果我不使用rails,我该如何设置时区? 最佳答案 ActiveRecord::Base.default_timezone='EasternTime(US&Canada)' 关于ruby-没有轨道的A

随机推荐