继承和组合

文章出处:

http://www.cnblogs.com/god_bless_you/archive/2011/02/17/1957009.html

http://blog.csdn.net/booboo2006/article/details/5824464

面向对象编程讲究的是代码复用,继承和组合都是代码复用的有效方法。组合是将其他类的对象作为成员使用,继承是子类可以使用父类的成员方法。

组合

  组合是通过对现有的对象进行拼装(组合)产生新的、更复杂的功能。因为在对象之间,各自的内部细节是不可见的,所以我们也说这种方式的代码复用是“黑盒式代码复用”。

优点:

  • 通过获取指向其它的具有相同类型的对象引用,可以在运行期间动态地定义(对象的)组合。
  • “黑盒”复用,被包含对象的内部细节对外是不可见。不破坏封装,整体类与局部类之间松耦合,彼此相对独立。
  • 整体类对局部类进行包装,封装局部类的接口,提供新的接口,具有较好的可扩展性。

缺点:

  • 整体类不能自动获得和局部类同样的接口,比继承实现需要的代码更多。
  • 不熟悉的代码的话,不易读懂。

继承

  在继承结构中,父类的内部细节对于子类是可见的。所以我们通常也可以说通过继承的代码复用是一种“白盒式代码复用”。

优点:

  • 简单易用,使用语法关键字即可轻易实现。
  • 易于修改或扩展那些父类被子类复用的实现。

缺点:

  • 编译阶段静态决定了层次结构,不能在运行期间进行改变。
  • 破坏了封装性,由于“白盒”复用,父类的内部细节对于子类而言通常是可见的。
  • 子类与父类之间紧密耦合,子类依赖于父类的实现,子类缺乏独立性。当父类的实现更改时,子类也不得不会随之更改。

两者的选择

is-a关系用继承表达,has-a关系用组合表达。继承体现的是一种专门化的概念而组合则是一种组装的概念。


二者间的区别何在,以及怎样在二者之间做出选择。

组合技术通常用于你想要在新类中使用现有类的功能而非它的接口的情形。即,你在新类中
嵌入某个对象,借其实现你所需要的功能,但新类的用户看到的只是你为新类所定义的接口,
而非嵌入对象的接口。为取得此效果,你需要在新类中嵌入一个private 的现有类的对象。

有时,允许类的用户直接访问新类中的组合成份是极具意义的;也就是说,将成员对象声明
为public 。如果成员对象自身都实现了具体实现的隐藏,那么这种做法就是安全的。当用户
能够了解到你在组装一组部件时,会使得端口更加易于理解。Car 对象即为一个好例子:

      

//: c06:Car.java 
        // Composition with public objects. 
          
        class Engine { 
          public void start() {} 
          public void rev() {} 
          public void stop() {} 
        } 
          
        class Wheel { 
          public void inflate(int psi) {} 
        } 
          
        class Window { 
          public void rollup() {} 
          public void rolldown() {} 
        } 
          
        class Door { 
          public Window window = new Window(); 
          public void open() {} 
          public void close() {} 
        } 
          
        public class Car { 
          public Engine engine = new Engine(); 
          public Wheel[] wheel = new Wheel[4]; 
          public Door 
            left = new Door(), 
            right = new Door(); // 2-door 
          public Car() { 
            for(int i = 0; i < 4; i++) 
              wheel[i] = new Wheel(); 
          } 
          public static void main(String[] args) { 
            Car car = new Car(); 
            car.left.window.rollup(); 
            car.wheel[0].inflate(72); 
          } 
      } ///:~

由于在这个例子中Car 的组合也是问题分析的一部分(而不仅仅是底层设计的一部分),所 

以使成员成为public 将有助于客户端程序员了解怎样去使用类,而且也降低了类开发者所面
临的代码复杂度。但务必要记得这仅仅是一个特例,一般情况下,你应使域(field)成为 private 。

在继承的时候,你会使用某个现有类,并开发一个它的特殊版本。通常,这意味着你在使用
一个通用性(general-purpose)的类,并为了某种特殊需要而将其特殊化。略微思考一下,
你就会发现,用一个“交通工具”对象来构成一部“车子”是毫无意义的,因为“车子”并
不包含“交通工具”,它仅是一种(is-a)交通工具。其中“is-a (是一个)”的关系是用继承
来表达的,而“has-a        (有一个)”的关系则是用组合来表达的



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值