使用模板模拟虚函数

翻译|其它|编辑:郝浩|2004-11-15 14:16:00.000|阅读 1753 次

概述:

# 界面/图表报表/文档/IDE等千款热门软控件火热销售中 >>


先看一个简单的虚函数的例子

#include <iostream>
using namespace std;
class B
{
public:
void Fun() { vf(); }
virtual void vf() { cout << "B::vf" << endl; }
};
class D : public B
{
public:
void vf() { cout << "D::vf" << endl; }
};

int main()
{
D d;
d.Fun();
B* pd = NULL;
pd = &d;
pd->Fun();
}

它的输出也很简单, 没有什么可以解释的。

D::vf

D::vf


我们的目标是使用模板来重新实现它,下面便是具体实现:

#include <iostream>
using namespace std;
template<typename T>
class Base
{
public:
void Fun()
{
T* pT = static_cast<T*>(this);
pT->vf();
}

void vf()
{
cout << "Base::vf" << endl;
}
};


class Derived : public Base<Derived>
{
public:
void vf()
{
cout << "Drived:vf" << endl;
}
};
int main()
{
Derived derived;
derived.Fun();
}

它的输出:

D::vf

我们的模板类真的实现了虚函数呀!其中关键的两条语句是: 
class Derived : public Base<Derived>
T* pT = static_cast<T*>(this);

利用模板的好处是很明显的:首先是编译后的程序变小了,省去了vptr 和 vtable。
第二时效率,模板类在编译期完成了静态绑定,比起虚函数的执行期动态绑定,节省了对於指针的指针的调用。 

如果模板真的那么强大,我想早就没有人用虚函数了。下面我们有必要谈一谈它的缺
点:

我们重新来改写main()函数
int main()
{
Derived derived;
derived.Fun();
Base *pBase = &derived;
pBase->Fun();


}
似乎没有问题,我们重新编译执行.......................报错了没有?报错的原因在於Base Class 后的尖括号,你必须声明那个<typename T>。我们重新改写一下

int main()
{
Derived derived;
derived.Fun();

Base<Derived> *pBase = &derived;
pBase->Fun();

}

重新编译执行,如果再有问题。那就是你自己的了。

模板是支持缺省值的,因此我们可以适当的偷下懒。将我们的基类改写一下。


template<type name T=Derived>
class Base
{
public:
void Fun()
{
T* pT = static_cast<T*>(this);
pT->vf();
}

void vf()
{
cout << "Base::vf" << endl;
}
};

但即便如此,尖括号也是必须的。


int main()
{
Derived derived;
derived.Fun();
Base<> *pBase = &derived;
pBase->Fun();

}


标签:

本站文章除注明转载外,均为本站原创或翻译。欢迎任何形式的转载,但请务必注明出处、不得修改原文相关链接,如果存在内容上的异议请邮件反馈至chenjj@evget.com


为你推荐

  • 推荐视频
  • 推荐活动
  • 推荐产品
  • 推荐文章
  • 慧都慧问
扫码咨询


添加微信 立即咨询

电话咨询

客服热线
023-68661681

TOP