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

woaidongmao

文章均收錄自他人博客,但不喜標(biāo)題前加-[轉(zhuǎn)貼],因其丑陋,見諒!~
隨筆 - 1469, 文章 - 0, 評(píng)論 - 661, 引用 - 0
數(shù)據(jù)加載中……

利用C++模板,代替虛函數(shù),實(shí)現(xiàn)類的靜態(tài)多態(tài)性(加入性能測(cè)試部分)

    熟悉模板編程的朋友或許聽到過(guò)這個(gè)技巧或者模式:Barton-Nackmann 技巧或者稱 奇異循環(huán)模板模式(Curiously Recurring Template Prattern)。
    
其實(shí)在 c++ 編程語(yǔ)言》這本bible 書里,在模板那章提到過(guò)一個(gè)很奇妙的類的實(shí)現(xiàn),用的就是這個(gè)技術(shù)。當(dāng)時(shí),我就被C++模板技術(shù)嘆為觀止。近期在學(xué)boost庫(kù)時(shí)偶然碰到了這個(gè)技巧,同時(shí)在寫一個(gè)類時(shí)引發(fā)了我的思考,這里就利用這個(gè)技巧來(lái)實(shí)現(xiàn),靜態(tài)多態(tài)函數(shù)(我自己發(fā)明的叫法,呵呵)。
 
我們知道C++的多態(tài)函數(shù)會(huì)帶來(lái)很多靈活性,但是不可避免的它是有運(yùn)行時(shí)的性能損失的。 c++的另一個(gè)強(qiáng)大特性就是模板了。模板給C++帶來(lái)了,編譯時(shí)的多態(tài),通過(guò)模板元編程,C++可以實(shí)現(xiàn)類似C#javarefection的特性。這里我就舉來(lái)實(shí)現(xiàn)利用模板來(lái)代替虛函數(shù)。
 
例子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();
}

  這是一個(gè)最普通的多態(tài)例子,下面看看一個(gè)比較有意思的例子:
例子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();
}

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

下面是這篇文章的重頭戲了,也是本文的目的所在。
我們看到例子12都采用虛函數(shù)來(lái)實(shí)現(xiàn)多態(tài),這個(gè)是一般選擇,如何用模板來(lái)實(shí)現(xiàn)多態(tài),而不是虛函數(shù)呢?
看這個(gè)例子:

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[])
{
    //
模板實(shí)現(xiàn)虛函數(shù)多態(tài)
    base<der> * pb= new der(100);
    pb->print();
    pb->m_print();

    //
動(dòng)態(tài)繼承
    der2<tmpclass> d2;
    d2.print();
    d2.m_print();
    d2.test();

    return 0;
}

哈哈,看class der是不是同樣實(shí)現(xiàn)了多態(tài)??而且語(yǔ)義和虛函數(shù)一致。可以進(jìn)一步提取downcast()部分到一個(gè)基類實(shí)現(xiàn)更普遍的寫法。

后面我實(shí)現(xiàn)了一個(gè)動(dòng)態(tài)繼承的類der2,它同樣提供了靈活的繼承用法,可惜好像因編譯器而定,在vc6環(huán)境下是不能通過(guò)編譯的,而在g++下是ok的。

 下面我編寫了一個(gè)性能測(cè)試?yán)虂?lái)測(cè)試?yán)锰摂M函數(shù)實(shí)現(xiàn)多態(tài)和模板實(shí)現(xiàn)多態(tài)的性能。代碼如下

#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;
}

我編譯了兩個(gè)版本一個(gè)優(yōu)化版本一個(gè)未優(yōu)化版本,運(yùn)行測(cè)試結(jié)果讓我有點(diǎn)意外:

這是未優(yōu)化版本的,結(jié)果顯示這兩種方法不相上下,虛函數(shù)還略優(yōu),~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

運(yùn)行優(yōu)化版本,性能優(yōu)勢(shì)一下體現(xiàn)出來(lái)了,模板實(shí)現(xiàn)是虛函數(shù)的十倍:

./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

有點(diǎn)驚人是不是?這個(gè)差別就是因?yàn)樘摵瘮?shù)是不可優(yōu)化和內(nèi)聯(lián)的,而模板函數(shù)是可內(nèi)聯(lián)的,這個(gè)性能差異就很大,再次隨著虛表的增大虛函數(shù)的調(diào)用是有性能退化的,而這點(diǎn)對(duì)于模板函數(shù)來(lái)說(shuō)是沒(méi)有的,因?yàn)樵诰幾g時(shí),這一切都是靜態(tài)了。不過(guò)客觀的說(shuō),虛函數(shù)多態(tài)是C++語(yǔ)言內(nèi)置的,在復(fù)雜度方面,應(yīng)該首選虛函數(shù),這里提供的這個(gè)方法只是作者在學(xué)習(xí)過(guò)程中的一個(gè)體會(huì),模板的世界實(shí)在是太奇妙和高深了。

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

評(píng)論

# re: 利用C++模板,代替虛函數(shù),實(shí)現(xiàn)類的靜態(tài)多態(tài)性(加入性能測(cè)試部分)  回復(fù)  更多評(píng)論   

“因?yàn)樵诰幾g時(shí),這一切都是靜態(tài)了;在復(fù)雜度方面,應(yīng)該首選虛函數(shù);”

虛函數(shù)的動(dòng)態(tài)特性的主要威力是大型項(xiàng)目中,模塊兒之間的可插拔
2010-08-28 12:04 | 路人

# re: 利用C++模板,代替虛函數(shù),實(shí)現(xiàn)類的靜態(tài)多態(tài)性(加入性能測(cè)試部分)  回復(fù)  更多評(píng)論   

哥運(yùn)行的未優(yōu)化版本可以看出,虛函數(shù)的優(yōu)勢(shì)還是很明顯的。。。\

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++模板,代替虛函數(shù),實(shí)現(xiàn)類的靜態(tài)多態(tài)性(加入性能測(cè)試部分)  回復(fù)  更多評(píng)論   

請(qǐng)問(wèn)一下怎么優(yōu)化?
2013-11-25 16:30 | Mr.Raindrop

# re: 利用C++模板,代替虛函數(shù),實(shí)現(xiàn)類的靜態(tài)多態(tài)性(加入性能測(cè)試部分)  回復(fù)  更多評(píng)論   

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

# re: 利用C++模板,代替虛函數(shù),實(shí)現(xiàn)類的靜態(tài)多態(tài)性(加入性能測(cè)試部分)[未登錄](méi)  回復(fù)  更多評(píng)論   

求問(wèn)怎么優(yōu)化的
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>
            久久久噜噜噜久久人人看| 国产一区二区精品久久91| 欧美福利视频网站| 亚洲欧美视频在线观看| 亚洲精品乱码久久久久久久久| 国产主播精品在线| 国产亚洲综合精品| 国产一区二区三区观看| 国产一区二区三区四区| 韩国精品久久久999| 在线播放日韩欧美| 日韩亚洲欧美成人一区| 99精品欧美一区二区三区| 日韩视频免费观看| 午夜激情一区| 欧美激情欧美狂野欧美精品| 亚洲美女黄网| 久久福利毛片| 9l视频自拍蝌蚪9l视频成人| 亚洲精品少妇| 亚洲专区在线| 欧美成人午夜视频| 欧美在线观看一区二区| 欧美成人午夜视频| 国产性天天综合网| 亚洲一区二区三区高清不卡| 久久精品日产第一区二区| 亚洲激情偷拍| 久久免费高清| 国产真实乱偷精品视频免| 在线一区二区视频| 亚洲人成人一区二区在线观看| 亚洲欧美日韩第一区| 欧美精品情趣视频| 亚洲第一色在线| 久久躁狠狠躁夜夜爽| 亚洲在线国产日韩欧美| 欧美日韩中文字幕综合视频| 亚洲黄色在线观看| 美女精品在线| 久久久www成人免费无遮挡大片 | 一区二区欧美亚洲| 猛男gaygay欧美视频| 精品999在线播放| 久久免费偷拍视频| 男同欧美伦乱| 亚洲免费在线精品一区| 国产欧美日韩免费| 午夜欧美精品| 午夜精品国产| 亚洲三级影院| 久久午夜av| 蜜臀91精品一区二区三区| 国产九九精品| 性做久久久久久| 久久亚洲免费| 1000部国产精品成人观看| 91久久精品国产91性色| 国产精品一级二级三级| 亚洲精品乱码久久久久久久久| 韩日欧美一区二区| 香蕉av777xxx色综合一区| 日韩亚洲欧美高清| 亚洲欧美激情四射在线日| 亚洲国产精品激情在线观看| 亚洲欧美视频一区二区三区| 亚洲亚洲精品在线观看| 麻豆精品视频在线观看视频| 久久激情视频久久| 国产精品午夜久久| 亚洲女人小视频在线观看| 亚洲一区在线视频| 国产精品久久久久久户外露出| 久久视频一区| 国产日本欧美视频| 久久一区二区三区国产精品| 麻豆精品在线观看| 亚洲电影免费在线| 男男成人高潮片免费网站| 麻豆亚洲精品| 在线不卡亚洲| 老司机精品视频一区二区三区| 久久综合999| 亚洲美女黄色片| 国产精品超碰97尤物18| 亚洲午夜久久久久久尤物| 午夜一区二区三区不卡视频| 久久综合久色欧美综合狠狠| 欧美性猛交xxxx乱大交退制版| 亚洲精品影视| 午夜在线成人av| 在线成人激情黄色| 欧美日韩国产bt| 欧美在线观看日本一区| 亚洲午夜未删减在线观看| 99国产精品私拍| 亚洲永久精品国产| 久久久久久网| 羞羞答答国产精品www一本| 亚洲国产专区校园欧美| 牛夜精品久久久久久久99黑人 | 亚洲精品国久久99热| 亚洲一级在线| 在线观看一区视频| 国产视频欧美| 国产亚洲精品美女| 国产精品久久久久久久久久免费| 久久综合九色| 久久精品导航| 欧美在线免费观看| 香蕉久久一区二区不卡无毒影院 | 亚洲欧洲一区二区三区| 免费观看一区| 一本色道久久综合狠狠躁篇怎么玩| 一区二区三区欧美日韩| 午夜精品久久久久久久99樱桃| 久久精品国产久精国产思思| 欧美电影在线观看完整版| av成人福利| 亚洲在线视频免费观看| 欧美高清一区| 欧美一级久久久久久久大片| 在线中文字幕不卡| 国产日韩欧美日韩大片| 欧美freesex8一10精品| 亚洲欧美日韩国产一区二区| 久久久亚洲成人| 中文精品一区二区三区| 国产伊人精品| 欧美经典一区二区| 久久xxxx精品视频| 亚洲图片欧洲图片日韩av| 亚洲第一区在线观看| 亚洲天堂av图片| 99精品视频一区二区三区| 亚洲第一中文字幕在线观看| 亚洲视频在线一区观看| 欧美激情影院| 美女91精品| 欧美亚洲三区| 欧美在线资源| 亚洲欧洲一区二区天堂久久 | 亚洲欧美日韩精品久久久久| 亚洲国产婷婷香蕉久久久久久| 亚洲男人的天堂在线aⅴ视频| 欧美国产大片| 亚洲美女91| 亚洲午夜激情网页| 亚洲色图制服丝袜| 久久综合给合| 亚洲午夜精品久久久久久app| 久久久在线视频| 亚洲精品系列| 中国av一区| 国产精品黄视频| 亚洲一区二区成人| 久久黄色小说| 亚洲第一色中文字幕| 另类亚洲自拍| 亚洲精品资源| 亚洲国产精品久久| 欧美精品国产一区二区| 亚洲精品视频一区| 性色av一区二区怡红| 国产一区av在线| 久久久综合视频| 亚洲三级色网| 亚洲一区二区三区中文字幕| 国产精品中文字幕欧美| 夜夜嗨av色综合久久久综合网| 午夜在线不卡| 亚洲国产日韩精品| 国产精品sm| 久久偷窥视频| 中国成人在线视频| 久久综合中文| 亚洲一区二区三区免费在线观看| 国产欧美日韩视频在线观看| 老色鬼精品视频在线观看播放| 亚洲美女尤物影院| 美脚丝袜一区二区三区在线观看| 99国产精品久久久久久久成人热| 国产精品久久久爽爽爽麻豆色哟哟| 欧美有码在线观看视频| 亚洲老司机av| 美国三级日本三级久久99| 午夜激情综合网| 91久久综合亚洲鲁鲁五月天| 国产老女人精品毛片久久| 久久一日本道色综合久久| 亚洲视频一区二区免费在线观看| 欧美18av| 久久久水蜜桃| 性欧美xxxx大乳国产app| 一区二区av| 亚洲欧洲综合另类在线| 国产一区二区精品久久91| 国产精品成av人在线视午夜片| 欧美成人午夜激情在线| 久久国产福利|