本文主要说明Java中继承与组合的概念,以及它们之间的联系与区别。首先文章会给出一小段代码示例,用于展示到底什么是继承。然后演示如何通过“组合”来改进这种继承的设计机制。最后总结这两者的应用场景,即到底应该选择继承还是组合。

1、继承

假设我们有一个名为Insect(昆虫)的类,这个类包含两个方法:1)移动move(); 2)攻击attack()。
代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
class Insect {
    private int size;
    private String color;
    public Insect(int size, String color) {
        this.size = size;
        this.color = color;
    }
    public int getSize() {
        return size;
    }
    public void setSize(int size) {
        this.size = size;
    }
    public String getColor() {
        return color;
    }
    public void setColor(String color) {
        this.color = color;
    }
    public void move() {
        System.out.println("Move");
    }
    public void attack() {
        move();  //假设昆虫在攻击前必须要先移动一次
        System.out.println("Attack");
    }
}

现在,你想要定义一个名为Bee(蜜蜂)的类。Bee(蜜蜂)是Insect(昆虫)的一种,但实现了不同于Insect(昆虫)的attack()和move方法。这时候我们可以用继承的设计机制来实现Bee类,就像下面的代码一样:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
class Bee extends Insect {
    public Bee(int size, String color) {
        super(size, color);
    }
    public void move() {
        System.out.println("Fly");
    }
    public void attack() {
        move();
        super.attack();
    }
}

1
2
3
4
5
6
public class InheritanceVSComposition {
    public static void main(String[] args) {
        Insect i = new Bee(1, "red");
        i.attack();
    }
}

InheritanceVSComposition作为一个测试类,在其main方法中生成了一个Bee类的实例,并赋值给Insect类型的引用变量 i。所以调用i的attack方法时,对应的是Bee类实例的attack方法,也就是调用了Bee类的attack方法。

类的继承结构图如下,非常简单:

输出:

1
2
3
Fly
Fly
Attack

Fly被打印了两次,也就是说move方法被调用了两次。但按理来讲,move方法只应当被调用一次,因为无论是昆虫还是蜜蜂,一次攻击前只移动一次。

问题出在子类(即Bee类)的attack方法的重载代码中,也就是super.attack()这一句。因为在父类(即Insect类)中,调用 attack方法时会先调用move方法,所以当子类(Bee)调用super.attack()时,相当于也同时调用了被重载的move方法(注意是子 类被重载的move方法,而不是父类的move方法)。

为了解决这个问题,我们可以采取以下办法:

  1. 删除子类的attack方法。这么做会使得子类的attack方法的实现完全依赖于父类对于该方法的实现(因为子类继承了父类的attack方法)。如果 父类的attack方法不受控制而产生了变更。比如说,父类的attack方法中调用了另外的move方法,那么子类的attack方法也会产生相应的变 化,这是一种很糟糕的封装。
  2. 也可以重写子类的attack方法,像下面这样:
1
2
3
4
public void attack() {
    move();
    System.out.println("Attack");
}

这样保证了结果的正确性,因为子类的attack方法不再依赖于父类。但是,子类attack方法的代码与父类产生了重复(重复的attack方法会使得很多事情变得复杂,不仅仅是多打印了一条输出语句)。所以第二种办法也不行,它不符合软件工程中关于重用的思想。

如此看来,继承机制是有缺点的:子类依赖于父类的实现细节,如果父类产生了变更,子类的后果将不堪设想。

2、组合

在上面的例子中,可以用组合的机制来替代继承。我们先看一下运用组合如何实现。

attack这一功能不再是一个方法,而是被抽象为一个接口。

1
2
3
4
interface Attack {
    public void move();
    public void attack();
}

通过对Attack接口的实现,就可以在实现类当中定义不同类型的attack。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
class AttackImpl implements Attack {
    private String move;
    private String attack;
    public AttackImpl(String move, String attack) {
        this.move = move;
        this.attack = attack;
    }
    @Override
    public void move() {
        System.out.println(move);
    }
    @Override
    public void attack() {
        move();
        System.out.println(attack);
    }
}

因为attack功能已经被抽象为一个接口,所以Insect类不再需要有attack方法。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
class Insect {
    private int size;
    private String color;
    public Insect(int size, String color) {
        this.size = size;
        this.color = color;
    }
    public int getSize() {
        return size;
    }
    public void setSize(int size) {
        this.size = size;
    }
    public String getColor() {
        return color;
    }
    public void setColor(String color) {
        this.color = color;
    }
}

Bee类一种Insect类,它具有attack的功能,所以它实现了attack接口:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
// 这个封装类封装了一个Attack类型的对象
class Bee extends Insect implements Attack {
    private Attack attack;
    public Bee(int size, String color, Attack attack) {
        super(size, color);
        this.attack = attack;
    }
    public void move() {
        attack.move();
    }
    public void attack() {
        attack.attack();
    }
}

类图:

测试类代码,将AttackImpl的实例作为Attack类型的参数传给Bee类的构造函数:

1
2
3
4
5
6
7
8
9
10
11
12
public class InheritanceVSComposition2 {
    public static void main(String[] args) {
        Bee a = new Bee(1, "black", new AttackImpl("fly", "move"));
        a.attack();
        // if you need another implementation of move()
        // there is no need to change Insect, we can quickly use new method to attack
        Bee b = new Bee(1, "black", new AttackImpl("fly", "sting"));
        b.attack();
    }
}

1
2
3
4
fly
move
fly
sting

3、什么时候该用继承,什么时候该用组合?

以下两条原则说明了应该如何选择继承与组合:

  • 如果存在一种IS-A的关系(比如Bee“是一个”Insect),并且一个类需要向另一个类暴露所有的方法接口,那么更应该用继承的机制。
  • 如果存在一种HAS-A的关系(比如Bee“有一个”attack功能),那么更应该运用组合。

总结来说,继承和组合都有他们的用处。只有充分理解各对象和功能之间的关系,才能充分发挥这两种机制各自的优点。

参考:

  1. Bloch, Joshua. Effective Java. Pearson Education India, 2008.
  2. http://stackoverflow.com/questions/49002/prefer-composition-over-inheritance
  3. http://www.javaworld.com/article/2076814/core-java/inheritance-versus-composition–which-one-should-you-choose-.html

Java中的继承与组合相关推荐

  1. 详细分析如何在java代码中使用继承和组合

    文章目录 继承与组合 何时在Java中使用继承 何时在Java中使用组合 继承与组成:两个例子 用Java继承重写方法 Java不具有多重继承 使用super访问父类方法 构造函数与继承一起使用 类型 ...

  2. java 的继承_关于java中的继承

    我们都知道Java中的继承是复用代码.扩展子类的一种方式,继承使得Java中重复的代码能够被提取出来供子类共用,对于Java程序的性能以及修改和扩展有很大的意义,所以这是一个非常重要的知识点. 那么对 ...

  3. java中抽象类继承抽象类_用Java中的抽象类扩展抽象类

    java中抽象类继承抽象类 示例问题 当我创建Java :: Geci抽象类AbstractFieldsGenerator和AbstractFilteredFieldsGenerator我遇到了一个不 ...

  4. java中抽象类继承抽象类_Java中的抽象类用示例解释

    java中抽象类继承抽象类 Abstract classes are classes declared with abstract. They can be subclassed or extende ...

  5. java中的继承(一)

    在java中,所有的类都有一个默认的父类Object,即使我们没有显式的去继承这个类.在Object类中定义了一些方法,其中有几个方法是我们必须掌握的,比如equals方法,hashCode方法,wa ...

  6. 习题 11.10 将本章11.8节中的程序片段加以补充完善,成为一个完整的程序。在程序中使用继承和组合。在定义Professor类对象prof1时给出所有数据的初值,然后修改prof1的生日数据。。。

    C++程序设计(第三版) 谭浩强 习题11.10 个人设计 习题 11.10 将本章11.8节中的程序片段加以补充完善,成为一个完整的程序.在程序中使用继承和组合.在定义Professor类对象pro ...

  7. Java中 利用继承的思想实现动物的分类,将动物分为水生动物和陆生动物,有其动物的属性和特征,又有其属性和特征,编程模拟。

    Java中 利用继承的思想实现动物的分类,将动物分为水生动物和陆生动物,有其动物的属性和特征,又有其属性和特征,编程模拟. 前言 动物在拥有自己本身所具有的特征的同时,还应该具有所在大类所具有的动物特 ...

  8. Java中的继承 与 多态(中)

    先导: 我们在<Java中的继承 与 多态(上)>当中讲解了如下几个问题,  1.继承是什么   2.super关键字   3.特殊考点-父子类中不同代码块的实现顺序 所以现在我们对于继承 ...

  9. 解析java中的继承

    解析java中的继承 1 含义 在符合当代社会的约定俗成的逻辑的情况下,很多类具有相同的属性和方法,我们这些共有的属性和方法抽象成一个类后,通过extends关键字使得很多类与抽象成的类建立父子关系, ...

最新文章

  1. gen_circle_contour_xld创建圆或圆弧的XLD轮廓
  2. Visual Studio将原生支持WSL 2
  3. python交叉验证函数_python – 在sklearn中使用交叉验证和AUC-ROC作为逻辑回归模型...
  4. 学了这几招,社交时人人都对你刮目相看(转载)
  5. [swift] LeetCode 136. Single Number
  6. 使用tail和head读取字节流
  7. python变量_Python变量
  8. Linux期末考试试题长沙理工,Linux 期末考试试题4
  9. Javascript多叉树的递归遍历和非递归遍历
  10. Coverity代码扫描工具
  11. html手机端的图片阅读器,兼容移动端与电脑端的图片查看器viewer.js插件
  12. k8s pod 详述
  13. 浅析大规模DDOS防御架构-应对T级攻防
  14. python画圣诞树【方块圣诞树、线条圣诞树、豪华圣诞树】
  15. 百度智能云章淼:详解企业级七层负载均衡开源软件BFE
  16. NBA表格_双红!34中24!NBA季后赛【掘金vs湖人】
  17. 鲜花电商狼烟四起,花点时间的“悦己”生意还能香多久?
  18. 深度学习“四大名著”发布!Python、TensorFlow、机器学习、深度学习四件套
  19. 微信小程序自定义组件中获取app实例的值,生命周期与页面的生命周期执行先后顺序
  20. wintogo详细过程记录

热门文章

  1. java要频繁调用容器时_Java知识点梳理
  2. 用java解决约瑟夫循环问题_Java采用循环链表结构求解约瑟夫问题
  3. wshttpbinding java_WCF自定义用户账号密码之WCF系结模式wsHttpBinding的Java调用
  4. vue项目目录结构分析、过滤器、vue文件中基础template、script、style
  5. JavaWeb三大组件(ServletFilterListener)
  6. 笔记本上的小键盘计算机怎样用,笔记本数字小键盘不能用怎么办【解决方法】...
  7. 谁是对象谁是类?你搞清楚了吗?
  8. 管理学习(2)——职场中最重要的四件事
  9. 2021消费者数智化运营白皮书
  10. 了不起的女子力:美妆消费蓝海与趋势赛道