Java基础:一、伴随多态的可互换对象(7)

  • 2020 年 3 月 17 日
  • 筆記

泛化

在处理类型的层次结构时,经常把一个对象不当作它所属的特定类型来对象,而是将其当作其基类的对象类对象。这称为“泛化” ,这样可以编写出不依赖特定类型的代码。

后期绑定

这将会产生一个问题,当我们将导出类型的对象当作其泛化基类型来看待时,比如我们自行车看作是交通工具,那么仍然存在一个问题,编译器在编译时是不可能知道自己应该执行哪一段代码将被执行,那对象如何会根据自身的具体类型来执行恰当的代码?

举个例子,在下面的图中,BirdController对象仅仅处理泛化的Bird对象,而不了解它们的确切类型。从BirdController的角度看,这么做非常方便,因为不需要编写特别的代码来判定要处理的Bird对象的确切类型或其行为。当move()方法被调用时,即便忽略Bird的具体类型,也会产生正确的行为(Goose(鹅)会走、非或游泳,Penguin(企鹅)走或游泳),这是如何发生的呢?

为了解决这个问题,面向对象程序设计语言使用了后期绑定的概念。当向对象发送消息时,被调用的代码直到运行时才能确定。编译器确保被调用方法的存在,并对调用参数和返回值执行类型检查,但是并不知道将被执行的确切代码。

为了执行后期绑定,Java使用一小段特殊的代码替代绝对地址调用。这段代码使用在对象存储的信息来计算方法体的地址。这样,根据这一小段代码的内容,每个对象都可以具有不同的行为表现,当向一个对象发送消息时,该对象就能够直到对这条消息应该做些什么。在Java 中,动态绑定是默认行为,不需要添加额外的关键字来实现多态(C++是使用virthal关键字来声明某个方法以实现后期绑定属性所带来的灵活性)。

向上转型

// 编写一个方法,这个方法可以将任何Shape类型作为参数  void doSomthing(Shape shape){    shape.erase();    //...    shape.draw();  }
Circle circle = new Circle();  Triangle triangle = new Triangle();  Line line = new Line();  doSomthing(circle);  doSomthing(triangle);  doSomthing(line);

对doSomthing()的调用会自动地正确处理,而不管对象的确切类型,比如doSomthing(circle); 由于Circle可以被doSomething()看作是Shape,也就是说,doSomthing()可以发送给Shape的任何消息,Circle也可以接收。

因为发送基类的任何消息,导出类都可以接收,那么将导出类看作是它的基类的过程称为向上转型。