青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

woaidongmao

文章均收錄自他人博客,但不喜標題前加-[轉貼],因其丑陋,見諒!~
隨筆 - 1469, 文章 - 0, 評論 - 661, 引用 - 0
數據加載中……

利用C++模板,代替虛函數,實現類的靜態多態性(加入性能測試部分)

    熟悉模板編程的朋友或許聽到過這個技巧或者模式:Barton-Nackmann 技巧或者稱 奇異循環模板模式(Curiously Recurring Template Prattern)。
    
其實在 c++ 編程語言》這本bible 書里,在模板那章提到過一個很奇妙的類的實現,用的就是這個技術。當時,我就被C++模板技術嘆為觀止。近期在學boost庫時偶然碰到了這個技巧,同時在寫一個類時引發了我的思考,這里就利用這個技巧來實現,靜態多態函數(我自己發明的叫法,呵呵)。
 
我們知道C++的多態函數會帶來很多靈活性,但是不可避免的它是有運行時的性能損失的。 c++的另一個強大特性就是模板了。模板給C++帶來了,編譯時的多態,通過模板元編程,C++可以實現類似C#javarefection的特性。這里我就舉來實現利用模板來代替虛函數。
 
例子1

#include <iostream>
using namespace std;
 
class common_base
{
public:
  virtual void fun
()=0
};
class common_derive:public common_base
{
public:
  void fun
()
  { cout<<"in common_derive fun()"<<endl;
};
 
void main()
{
  common_base * pb = new common_derive;
  pb->fun();
}

  這是一個最普通的多態例子,下面看看一個比較有意思的例子:
例子2

template<typename T>
class class1
{
public:
    class1(T t):m_val(t){}
    virtual T getVal(){
        cout<<"in class1,val =="<< m_val <<endl;
        return m_val;
    }
private:
    T m_val;
};

class derived: public class1<int>
{
public:
    derived(int i):class1<int>(i){}
    int getVal()
    {
        cout<<"in derived"<<endl;
        return class1<int>::getVal();
    }
};

template<typename T>
class derived2: public class1<T>
{
public:
    derived2(T val):class1<T>(val){}
    T getVal()
    {
        cout<<"in derived2"<<endl;
        return class1<T>::getVal();
    }
};

void main()
{
    class1<int> * pbase = new derived(10);
    pbase->getVal();

    class1<int> * pb2 = new derived2<int>(10);
    pb2->getVal();
}

這個例子我的用意是說明:模板類的虛函數多態,而且派生類可以有兩種選擇,一個實現為普通類,繼承的是模板基類的特化類,一個就實現模板類(如 derived2)。很明顯模板繼承類有著普通類不可比擬的靈活性。

下面是這篇文章的重頭戲了,也是本文的目的所在。
我們看到例子12都采用虛函數來實現多態,這個是一般選擇,如何用模板來實現多態,而不是虛函數呢?
看這個例子:

template<class derive>
class base
{
public:
    void print()
    {
        derive::print();
    }
    void m_print()
    {
        downcast()->derive::m_print();
    }
protected:
    inline derive * downcast()
    {
        return static_cast<derive *>(this);
    };
    inline const derive * downcast()const
    {
        return static_cast<const derive *>(this);
    };
};

class der:public base<der>
{
public:
    der(int foo):_foo(foo){}
    static void print()
    {
        cout<<"in der print"<<endl;
    };
    void m_print()
    {
        cout<<"in der member fun m_print"<<endl;
        cout<<"has member foo="<<_foo<<endl;
    }
private:
    int _foo;
};

template<class base>
class der2:public base
{
public:
    static void print()
    {
        cout<<"in der2 print"<<endl;
    };
    void m_print()
    {
        cout<<"in der2 member fun m_print"<<endl;
    }
};

class tmpclass
{
public:
    void test()
    { cout<<"in test"<<endl;}
};

int main(int argc, char* argv[])
{
    //
模板實現虛函數多態
    base<der> * pb= new der(100);
    pb->print();
    pb->m_print();

    //
動態繼承
    der2<tmpclass> d2;
    d2.print();
    d2.m_print();
    d2.test();

    return 0;
}

哈哈,看class der是不是同樣實現了多態??而且語義和虛函數一致。可以進一步提取downcast()部分到一個基類實現更普遍的寫法。

后面我實現了一個動態繼承的類der2,它同樣提供了靈活的繼承用法,可惜好像因編譯器而定,在vc6環境下是不能通過編譯的,而在g++下是ok的。

 下面我編寫了一個性能測試例程來測試利用虛擬函數實現多態和模板實現多態的性能。代碼如下

#include <iostream>
using namespace std;
#include <sys/time.h>

class common_base
{
public:
    common_base(int iloop){_iloop=iloop;}
    virtual void virtual_fun()=0;
    void timesum_fun()
    {
        struct timeval begin,end;
        gettimeofday(&begin, NULL);
        for(int i=0;i<_iloop;i++)
            virtual_fun();
        gettimeofday(&end, NULL);
        cout<< "using time :" << end.tv_sec-begin.tv_sec + (end.tv_usec - begin.tv_usec)/1000000.0<<"  second"<<endl;
    };
private:
    int _iloop;
};
class common_derive:public common_base
{
public:
    common_derive(int iloop):common_base(iloop){_foo=0;}
    void virtual_fun()
    {
        ++_foo;
        --_foo;
    }
private:
    int _foo;
};

template<class derive>
class base
{
public:
    base(int iloop){_iloop=iloop;}
    void timesum_fun()
    {
        struct timeval begin,end;
        gettimeofday(&begin, NULL);
        for(int i=0;i<_iloop;i++)
            templ_fun();

        gettimeofday(&end, NULL);
        cout<< "using time :" << end.tv_sec-begin.tv_sec + (end.tv_usec - begin.tv_usec)/1000000.0<<"  second"<<endl;
    }
    inline void templ_fun()
    {
        downcast()->derive::templ_fun();
    }
protected:
    inline derive * downcast()
    {
        return static_cast<derive *>(this);
    };
    inline const derive * downcast()const
    {
        return static_cast<const derive *>(this);
    };
private:
    int _iloop;
};

class der:public base<der>
{
public:
    der(int iloop):base<der>(iloop){_foo=0;}
    inline void templ_fun()
    {
        ++_foo;
        --_foo;
    }
private:
    int _foo;
};

int main()
{
  int loop=1000*1000*100;
  common_base * pb = new common_derive(loop);
  base<der> * ptempb= new der(loop);
  for(int i =3;i-->0;)
  {
      cout<<"virtual function test: looptime="<<loop<<endl;
      pb->timesum_fun();
      cout<<"template function test: looptime="<<loop<<endl;
      ptempb->timesum_fun();
  }
  delete pb;
  delete ptempb;
  return 0;
}

我編譯了兩個版本一個優化版本一個未優化版本,運行測試結果讓我有點意外:

這是未優化版本的,結果顯示這兩種方法不相上下,虛函數還略優,~O

./cmp_test
virtual function test: looptime=100000000
using time :1.03824  second
template function test: looptime=100000000
using time :1.63043  second
virtual function test: looptime=100000000
using time :1.03768  second
template function test: looptime=100000000
using time :1.62773  second
virtual function test: looptime=100000000
using time :1.63104  second

運行優化版本,性能優勢一下體現出來了,模板實現是虛函數的十倍:

./cmp_test_optimize
virtual function test: looptime=100000000
using time :0.615542  second
template function test: looptime=100000000
using time :0.055584  second
virtual function test: looptime=100000000
using time :0.624778  second
template function test: looptime=100000000
using time :0.057419  second
virtual function test: looptime=100000000
using time :0.624977  second
template function test: looptime=100000000
using time :0.059442  second

有點驚人是不是?這個差別就是因為虛函數是不可優化和內聯的,而模板函數是可內聯的,這個性能差異就很大,再次隨著虛表的增大虛函數的調用是有性能退化的,而這點對于模板函數來說是沒有的,因為在編譯時,這一切都是靜態了。不過客觀的說,虛函數多態是C++語言內置的,在復雜度方面,應該首選虛函數,這里提供的這個方法只是作者在學習過程中的一個體會,模板的世界實在是太奇妙和高深了。

posted on 2008-05-22 22:41 肥仔 閱讀(7414) 評論(5)  編輯 收藏 引用 所屬分類: C++ 基礎

評論

# re: 利用C++模板,代替虛函數,實現類的靜態多態性(加入性能測試部分)  回復  更多評論   

“因為在編譯時,這一切都是靜態了;在復雜度方面,應該首選虛函數;”

虛函數的動態特性的主要威力是大型項目中,模塊兒之間的可插拔
2010-08-28 12:04 | 路人

# re: 利用C++模板,代替虛函數,實現類的靜態多態性(加入性能測試部分)  回復  更多評論   

哥運行的未優化版本可以看出,虛函數的優勢還是很明顯的。。。\

virtual function test: looptime=100000000
using time :0.849799 second
template function test: looptime=100000000
using time :1.6305 second
virtual function test: looptime=100000000
using time :0.848855 second
template function test: looptime=100000000
using time :1.6228 second
virtual function test: looptime=100000000
using time :0.838681 second
template function test: looptime=100000000
using time :1.62295 second
2010-10-15 19:26 | sipo

# re: 利用C++模板,代替虛函數,實現類的靜態多態性(加入性能測試部分)  回復  更多評論   

請問一下怎么優化?
2013-11-25 16:30 | Mr.Raindrop

# re: 利用C++模板,代替虛函數,實現類的靜態多態性(加入性能測試部分)  回復  更多評論   

然并卵。
common_base * pb = new common_derive(loop);
// 虛函數的方式,pb的指針類型是一樣的,都是基類的指針。
base<der> * ptempb= new der(loop);
// 模板偽多態,這個ptempb指針類型都是不一樣的,不能用同一個指針指向不同子類的實例,也就失去了多態的意義。
2015-08-21 11:32 | ngugc

# re: 利用C++模板,代替虛函數,實現類的靜態多態性(加入性能測試部分)[未登錄]  回復  更多評論   

求問怎么優化的
2016-06-06 21:46 | hunter
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            夜夜嗨一区二区| 亚洲欧美在线x视频| 亚洲精品国久久99热| 尤物精品在线| 亚洲福利电影| 日韩亚洲国产欧美| 国产欧美在线观看| 麻豆成人91精品二区三区| 久久网站免费| 久久在线免费观看| 久久久女女女女999久久| 每日更新成人在线视频| 久久亚洲精品欧美| 亚洲国产精品123| 欧美激情亚洲另类| 亚洲精品欧洲精品| 亚洲免费网址| 免费永久网站黄欧美| 欧美日韩视频在线一区二区观看视频| 欧美精选午夜久久久乱码6080| 欧美日韩情趣电影| 国产一区二区黄色| 99成人在线| 久久九九99视频| 亚洲国产精品久久久久婷婷老年| 日韩亚洲欧美一区二区三区| 亚洲欧美综合v| 免费影视亚洲| 国产乱子伦一区二区三区国色天香| 激情婷婷欧美| 亚洲综合二区| 欧美福利电影在线观看| 在线视频你懂得一区| 久久久久看片| 国产精品国产福利国产秒拍| 亚洲国产成人午夜在线一区| 香蕉乱码成人久久天堂爱免费| 裸体歌舞表演一区二区 | 久久久久国产一区二区| 亚洲国产二区| 亚洲精品一二区| 午夜精品久久久久久久99黑人| 久久婷婷丁香| 亚洲天堂成人在线视频| 美女主播精品视频一二三四| 国产女主播在线一区二区| 亚洲精品中文字幕女同| 免费久久99精品国产| 亚洲欧美日韩综合国产aⅴ | 国产亚洲一区二区三区在线播放| 欧美激情一区二区三区在线视频| 国产免费成人| 午夜精品视频在线观看一区二区 | 久久久www免费人成黑人精品| 欧美剧在线免费观看网站| 男同欧美伦乱| 亚洲福利视频在线| 嫩草影视亚洲| 久久婷婷成人综合色| 国产一区二区三区精品欧美日韩一区二区三区 | 亚洲免费观看| 欧美成年视频| 亚洲精品美女在线观看| 男同欧美伦乱| 免费欧美电影| 亚洲毛片播放| 日韩亚洲欧美一区二区三区| 欧美日韩 国产精品| 一本色道久久综合一区| 亚洲日本中文字幕免费在线不卡| 牛牛影视久久网| 亚洲人成在线播放网站岛国| 欧美国产三区| 麻豆精品视频在线观看视频| 亚洲欧洲久久| 亚洲精品中文字幕在线观看| 欧美日韩精品国产| 亚洲专区国产精品| 午夜精品久久久久久久久久久久久 | 日韩视频精品在线| 欧美婷婷在线| 香蕉视频成人在线观看| 久久国产福利| 亚洲激情自拍| 99视频精品在线| 国产美女精品免费电影| 嫩草伊人久久精品少妇av杨幂| 欧美1区3d| 国产综合网站| 欧美成人自拍| 欧美激情一区二区三区不卡| 这里是久久伊人| 午夜精品久久久久久99热| 一区在线电影| 亚洲欧洲综合另类| 国产精品红桃| 欧美大片在线看| 欧美日韩中文字幕| 久久亚洲欧美国产精品乐播| 欧美激情一区二区久久久| 羞羞视频在线观看欧美| 你懂的视频欧美| 欧美一区二区视频网站| 免费欧美日韩| 欧美中文字幕在线视频| 欧美激情综合色| 久久久久久久999精品视频| 欧美精品成人一区二区在线观看| 亚洲视频狠狠| 久久一综合视频| 亚洲自拍电影| 欧美风情在线| 久久这里有精品视频| 国产精品美女久久久久久免费 | 一本久道久久久| 久久久久久久性| 国产视频亚洲| 亚洲黑丝一区二区| 韩国视频理论视频久久| 一区二区三区欧美亚洲| 亚洲日本中文字幕| 亚洲国产婷婷| 亚洲大胆在线| 久久亚洲综合色| 久久午夜色播影院免费高清| 国产精品伦一区| 99成人精品| 在线亚洲一区观看| 欧美精品一区二区精品网| 久久综合影音| 国内免费精品永久在线视频| 亚洲综合导航| 午夜精品一区二区在线观看 | 午夜精品久久久久久久99热浪潮| 国产精品a久久久久| 亚洲欧洲日韩女同| 亚洲国产精品一区制服丝袜| 久久精品视频在线看| 欧美在线观看日本一区| 国产精品视频观看| 亚洲欧美在线免费| 久久精品国产综合| 国产欧美精品一区aⅴ影院| 亚洲一区二区在线免费观看视频 | 久久激情婷婷| 国产综合一区二区| 久久琪琪电影院| 欧美大秀在线观看| 日韩手机在线导航| 欧美视频一区二区三区四区| 99视频在线观看一区三区| 亚洲视频在线一区观看| 国产精品va在线| 亚洲欧美日韩精品久久亚洲区| 欧美在线影院| 在线观看亚洲专区| 麻豆精品视频在线观看视频| 亚洲韩国青草视频| 亚洲一区不卡| 国产午夜精品久久久久久久| 久久久久久久久久久久久9999| 欧美高清视频一区二区| 日韩网站免费观看| 国产麻豆视频精品| 久久精品道一区二区三区| 欧美福利在线观看| 亚洲乱码国产乱码精品精| 欧美四级伦理在线| 欧美专区日韩视频| 亚洲国产一二三| 欧美亚洲综合在线| 亚洲福利在线观看| 国产精品免费小视频| 鲁大师成人一区二区三区| av不卡免费看| 玖玖玖国产精品| 亚洲欧美激情一区| 亚洲欧洲三级电影| 国产亚洲欧美色| 欧美日韩另类一区| 久久一区二区三区国产精品| 欧美一区二区三区免费在线看| 一区在线观看| 国产精品视频免费| 欧美激情一级片一区二区| 欧美一区二区三区四区在线观看地址 | 亚洲国产精品999| 国产欧美精品日韩| 欧美日韩hd| 免费日韩视频| 久久九九电影| 欧美亚洲综合在线| 亚洲亚洲精品在线观看| 日韩视频一区二区三区在线播放免费观看 | 亚洲欧美国产制服动漫| 精品99一区二区| 欧美日韩国产在线播放| 亚洲乱码精品一二三四区日韩在线| 久久免费视频在线观看| 99国产精品视频免费观看一公开 |