Chinaunix首页 | 论坛 | 博客
  • 博客访问: 988611
  • 博文数量: 158
  • 博客积分: 4380
  • 博客等级: 上校
  • 技术积分: 2367
  • 用 户 组: 普通用户
  • 注册时间: 2006-09-21 10:45
文章分类

全部博文(158)

文章存档

2012年(158)

我的朋友

分类: C/C++

2012-11-19 13:16:52

非多态的继承

由“什么情况下,基类可以不提供非虚析构函数?”而来,

1. 位继承。顾名思义,继承的是位,而不是接口
比如
struct Head
{
    int length;
    int flag;
};
struct MyFile : Head
{
    ……
};
[注]:位继承 和 包含 是有强烈语义差别的,虽然有可以互相变通的语法和语义。

2. 实现继承。顾名思义,继承的是实现,而不是接口
比如 protected继承 和 private继承。
[注]:继承实现应当比继承接口更安全也用得更广,看STL的实现可知,但很多人受XXXX荼毒太深滥用了面向对象。

3. 非多态全继承。顾名思义,继承实现也继承接口,但却是非多态的
比如我想在map的基础上增加一个功能变成字典
struct Dictionary : public map
{
    void foo() const;
};
[注]:map是一个不含任何虚函数的类。

以上三种情况下,从多态上讲(这个限定语是必须的)派生类都不是一个基类,因此你不可以使用dynamic_cast进行转化.

4. 在这一种情况下,设计者本来的意图也不是多态继承,但可惜从语法上说它是多态继承也能说得过去。(这种矛盾其实经常遇到,比如mutable的由来)
比如在第三段给的代码中,假设(只是假设,而不是事实)map是一个含有虚函数的类。这样一来从语法上可以把Dictionary看成一个map,不过既然设计者没有多态的意图,那么把它作多态来用的也只有傻子,当然它也就无须提供虚析构函数。

这四种情况都是非面向对象的继承。

阅读(3604) | 评论(20) | 转发(2) |
0

上一篇:互锁操作

下一篇:清除输入流的错误

给主人留下些什么吧!~~

网友评论2012-11-19 13:19:44

anonymous
实现继承应该改为用内聚(视其为成员变量), 我觉得尽量少用实现.

网友评论2012-11-19 13:19:30

pAnic
1,只要父类不是纯虚,你继承接口的同时也就继承了实现。只不过两者各占一部分。
2,面向对象的三要素就是 继承,封装,多态。少了一个就像三脚猫瘸了一条腿。

我也不是很清楚一个虚析构到底带来什么效率之外的副作用,不过对于小型对象来说,虚析构引起对象尺寸的变化可能是更需要考虑的问题。

网友评论2012-11-19 13:19:09

清风雨
语法上讲,以上情况可以不提供非虚析够器,学习了。

有几个疑问:
1.继承实现应当比继承接口更安全也用得更广?何知更安全?何晓更广?(我可以转型到父类,还是编译能通过;java中只有一种继承就是public的,我所见的几乎大多数C++代码都是public继承)
2.不具有多态就不是面向对象?何解?(如果面向对象一定要多态,那么C++可以象java那么,所有的函数都实际上是虚的,还可以少打个virtual。这个问题太难说,想了好久,不知道怎么表达。 )

个人是这样看非虚析构器的:
1.苛刻的考虑虚函数表指针的开销(基本上,我不会在意)
2.析够器里面没有任何资源的释放工作(如果子类需要被继承,可以把自己的析构器虚化)
3.不希望别人继承我的类(继承了可能会出错)

网友评论2012-11-19 13:18:55

周星星
不错,应该说“非面向对象的继承”,面向对象必须要有多态,不具有多态就不是面向对象。

网友评论2012-11-19 13:18:42

pAnic
似乎多态仅仅是指接口继承。