“瑜珈山夜话”--- 寻根究底谈“继承”(一) (转)

“瑜珈山夜话”--- 寻根究底谈“继承”(一) (转)[@more@]

  摘要:继承是C++的一个很重要的特性,也是OO的三大特征之一,希望对此做一个简单的论述,能消除你一些困惑。
 
  继承是什么?
  继承是将相关的类组织起来,并分亨其间的共通数据和操作行为的一种方法,同时也要注意到继承关系是一种强耦合的关系。
 
  继承的目的是什么?
  说到继承的目的,人们总是会想到代码重用,实则不然,代码重用只不过是继承的一个副作用,继承的主要目的是表达一个外部有意义的关系,该关系描述了问题域内的2个实体之间的行为关系。换句话说,继承是因问题域的现实性而产生的,并不是由于解域内的技术目的而出现的。

  继承的障碍是什么?
  继承的使用并不像我们想象的那么简单,在决定继承的时候,有很多语言特性会构成一定的障碍。
  1、非虚成员函数的存在。
  如果我们确定了一个基类中的某个成员函数是非虚的,那就意味着这个函数在派生类中不应该被重新定义,如果你重新定义了,所得的结果很可能不是你所期望的,例如:
  class A
  {  public: void f() { cout<  class B: public A
  {  public: void f() { cout<  A* pA=new B;
  pA->f();
  delete pA;
  这里,我们可能期望pA->f()会输出B::f,但是实际上是A::f,当然,如果把它声明为virtual就没有问题了,关键是我们怎么能够明确确定那个函数应该声明为virtual呢?如何使基类能够完全预测到子类的各种需求?毫无疑问,这是一个挑战!也许把所有的基类成员函数都声明为virtual是一个简单的解决办法,但是这样做会大大降低程序执行效率,对于如此注重效率的C++来说,这么做是对它的一个背叛,C++更希望我们只把那些需要重定义的函数声明为virtual。
  2、基类成员的过度保护
  封装是一个很好的特性,但是封装的度很难掌握,例如:
  class A
  {  private: class P { ...};  };
  class B : public A::P { ... };
  有经验的程序员马上就会意识到这是一个错误:无法获取A::P,因为它的权限是Private!当然这里只需要把private改为protected就可以了,但是问题的关键在于基类如何预测到子类需要继承的类究竟是什么?同上一个障碍一样,这也是一个挑战。天真的程序员可能以为只要把基类中所有的成员都声明为public/protected就万事大吉了,但是实际上如果我们的类发布之后,public/protected的成员就再也无法改变,否则势必会中断客户的代码,这就要求我们尽量把实现细节封装为private的,只把那些子类需要变动的成员声明为public/protected权限(虚函数可以声明为private的,这是一个例外),但是对基类的设计者要求如此之高,也是非常困难的。
  3、基类中模块化设计不足
  模块化会使程序更加简洁、有效,但是对于基类来说,要做到有效的模块化并不容易。例如我们有一个二分查找树BSTree,定义如下:
  template
  class BSTree
  {
  private:
  class Node
  {
  public:
  T t;
  Node* left;
  Node* right;
  Node(const T& _t):t(_t){ }
  ...
  };
  Node* root;
  ...
  public:
  void insert(const T& t);
  ...
  protected:
  virtual void doinsert(const T& t, Node*& n);
  ...
  };
  template
  void BSTree::doinsert(const T& t, Node*& n)
  {
  if(n==0) n=new Node(t);
  else
  {
  if(tt) doinsert(t, n->left);
  else doinsert(t, n->right);
  }
  }
  现在呢,我们要定义一个红黑树,定义如下:
  template
  class RBTree: public BSTree
  {
  protected:
  class Node: public BSTree::Node
  {
  public:
  bool is_red;
  Node(const T& t);
  };
  void doinsert(const T& t, BSTree::Node*& n);
 virtual void rebalance(Node* n);
  ...
  };
  template
  void BSTree::doinsert(const T& t, Node*& n)
  {
  if(n==0)
  {
  Node m=new Node(t);
  n=m;
  rebalance(m);
  }
  else
  {
  if(tt) doinsert(t, n->left);
  else doinsert(t, n->right);
  }
  }
  我们发现BSTree::doinsert和RBTree::doinsert代码大致相同,这就存在着复制代码操作,我们知道代码复制工作十分乏味、易出错、代码臃肿、维护困难...所以一个好的基类应该使派生类尽量少的复制代码,最好不复制。看看我们的基类:很多二分查找树都需要创建不同的节点,也有rebalance操作。好了,我们应该对基类BSTree作如下修改:
  Template
  class BSTree
  {
  protected:
  virtual Node* new_node(const T& t)
  { return new Node(t); }
  virtual void rebalance(Node* n) { }
  ...
  };
  这时候doinsert改动如下:
  template
  void BSTree::doinsert(const T& t, Node*& n)
  {
  if(n==0)
  {
  n=new_node(t);
  rebalance(n);
  }
  else
  {
  if(tt) doinsert(t, n->left);
  else  doinsert(t, n->right);
  }
  }
  这时候派生类RBTree定义改为:
  template
  class RBTree: public BSTree
  {
  protected:
  Node* new_node(const T& t)
  { return new Node(t); }
  void rebalance(BSTree::Node* n)
  {  ...  }
  ...
  };
  这样一来,程序员就无需复制代码了。我们发现,如果要使派生类的客户永远不复制代码,那么就要把派生类需要改变的代码分离出来,形成一个单独的模块函数(虚),但是在我们没有足够的派生类的信息的时候,这样做是不可能的,就算可能,难度也是相当得高,同时,大量的虚函数也会降低程序的执行效率。 
  4、friend关键字的过分使用
  这个问题的根源在于友员关系的不继承性。我们仍然用上面的例子,不过做一下变动:
  template class BSTree;
  template
  class BSNode
  {
  protected:
 T t;
  BSNode(const T& t);
  friend class BSTree;
  };
  template
  class BSTree
  {
  ...没有了nested Node类
  };
  这里,由于BSNode的实现属于BSTree的实现细节,同时为了防止BSNode派生类偶然存取BSNode的成员,所以我们把他的所有成员都声明为Protected,同时让BSTree称为它的友员。但是由于RBTree要存取BSNode的成员,再加上友员的非继承,使事情变得复杂起来,通常有2种办法解决这个问题:
  1、将BSNode的成员声明为public,但是这样一来friend也就没有什么意义了。
  2、在RBNode类中增加一个存取函数,但是和不用friend相比,麻烦多了。
  另外还有一些其它的抉择也是让人头疼,例如:基类中的成员变量过多,继承的属性选择等。

未完(待续...)


来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/10748419/viewspace-960904/,如需转载,请注明出处,否则将追究法律责任。

转载于:http://blog.itpub.net/10748419/viewspace-960904/

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值