Chinaunix首页 | 论坛 | 博客
  • 博客访问: 988111
  • 博文数量: 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,不过既然设计者没有多态的意图,那么把它作多态来用的也只有傻子,当然它也就无须提供虚析构函数。

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

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

上一篇:互锁操作

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

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

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

清风雨
1。《Effective C++》是一本很不错的C++指导书,虽然《More Effective C++》很烂。嗯?我觉得《M》蛮好的。看来,我去得看看《E》。顺便,期望你能给出它烂的理由。
2。你的2、3我就不再过多继续了,累。你说基于对象和面向对象的很同意。只是如果空头的说说实际意义不大,我是实用者,虽然也相信和重视学术。
3。不使用virtual得析构器的问题,记得我上面说了3种情况,我认为那时可以不virtual析构。

顺便pAnic的说法,我反对友元(这个,肯定有很多人要极力的反对我了)。当然,我不否认,很多时候,他确实很有用(记得goto也很有用,其实我不那么反对的,甚至自己也企图去用,不过后来写好了,来来去去删了几次,最后决定还是删除了goto)。知道很多时候,他可以增加模块内的内聚,只是我觉得这个模块本来可以好好的作为独立零件的,被强行一整个了,而且他所带来的好处又确实有限,所以,我不用它。在我的地盘,就尽量杜绝friend;不是的,就各持己见好了。

网友评论2012-11-19 13:22:02

Panic
析构函数私有,还有一种情况就是工厂对象,不希望在工厂外部删除对象的时候,把工厂类设为工厂对象的友元,工厂对象的析构和构造全部私有。

网友评论2012-11-19 13:21:46

周星星
to 清风雨:
1. 《Effective C++》是一本很不错的C++指导书,虽然《More Effective C++》很烂。
2. 关于面向对象的定义,请先查一下再来讨论,不要像某甲一样定义了与众不同的宇、宙、边、界来讨论空间物理。我的意思是:你也许有自己的面向对象的定义,但我这儿讲的面向对象是大家所承认的定义。
我猜想你是把“基于对象”也当成了“面向对象”的一部分。
3. 关于“默认所有成员函数都virtual”的问题,我再说一次(我希望是最后一次):C++不是一个仅面向对象的语言,面向对象也不是C++的主要部分。
由我的“面向对象必须要有多态”和“C++不是一个仅面向对象的语言”得不出C++应当默认所有成员函数都virtual的结论。只有“面向对象必须要有多态”和“C++仅是一个面向对象的语言”才能得出C++应当默认所有成员函数都virtual的结论,所以我在上一个回贴中指出了你的错误“你认为C++是Only面向对象的,或者主要是面向对象的,而实际上C++本不是。”。

网友评论2012-11-19 13:21:28

ilovevc
什么情况下,基类可以不提供非虚析构函数
至少还有一条:
protected 析构函数, 只作为基类被派生类使用, 一般client也delete不掉. (编译错), 因此不会出现析构不完全的问题.

当然private析构函数那就纯粹是找别扭, 估计也就singleton使用, 也无什么继承一说了.

网友评论2012-11-19 13:21:14

清风雨
不小心一回车,就发出去了。

如果面向对象一定要多态,那么C++可以象java那么,所有的函数都实际上是虚的,还可以少打个virtual。

就是说 如果你认为面向对象一定要多态的话,才说不如就默认所有成员函数都virtual。
事实是不是,也就是说面向对象不一定非要多态。(因为你说“面向对象必须要有多态,不具有多态就不是面向对象。”,我觉得说法不妥,所以,才这样说,确信你要表达的具体意思,和想知道为什么要这么说:是我理解错误呢?还是你确实表达欠妥?还是只是为了强调?)

就着些。

我只是看到了,想问问。俺不喜欢被带帽子的(我想其实大家都不喜欢的^_^),因为以前大学时有个同学给我扣帽子,搞得很烦,看到帽子时,还是有些不高兴的。不过还好也不要紧,事实还是事实。

关于安全和广泛,俺就不多说了。我只是要明确一下,并不是好像public是一件愚蠢的和不适合的做法。
protected、