« 深入浅出MFC-MFC六大关键技术笔记 | Home | 放假了,笔记暂缓 »
thinking in java 第六、七章
By robinhesky | 01月 21, 2008
1.类再生
Java引人注目的一项特性是代码的重复使用或者再生。但最具革命意义的是,除代码的复制和修改以外,我们还能做多得多的其他事情。”
在象C那样的程序化语言里,代码的重复使用早已可行,但效果不是特别显著。与Java的其他地方一样,这个方案解决的也是与类有关的问题。我们通过创建新类来重复使用代码,但却用不着重新创建,可以直接使用别人已建好并调试好的现成类。但这样做必须保证不会干扰原有的代码。
有两个达到这一目标的方法:
(1)在新类里简单地创建原有类的对象。我们把这种方法叫作“合成”,因为新类由现有类的对象合并而成。我们只是简单地重复利用代码的功能,而不是采用它的形式。
(2)创建一个新类,将其作为现有类的一个“类型”。我们可以原样采取现有类的形式,并在其中加入新代码,同时不会对现有的类产生影响。这种魔术般的行为叫作“继承”(Inheritance),涉及的大多数工作都是由编译器完成的
每种非基本类型的对象都有一个toString()方法。若编译器本来希望一个String,但却获得某个这样的对象,就会调用这个方法。所以在下面这个表达式中:
System.out.println(”source = ” + source) ;
编译器会发现我们试图向一个WaterSource添加一个String对象(”source =”)。这对它来说是不可接受的,因为我们只能将一个字串“添加”到另一个字串,所以它会说:“我要调用toString(),把source转换成字串!”经这样处理后,它就能编译两个字串,并将结果字串传递给一个System.out.println()。每次随同自己创建的一个类允许这种行为的时候,都只需要写一个toString()方法。
2.初始化的顺序
我个人的理解是,如果有派生的关系,先从根类开始初始化(其它类都由它派生出来),如果各个类中有static成员,那最初初始化的是static,再从根类开始初始化。而根类的初始化顺序应为先初始化成员,再调用构造函数进行初始化,其派生类也依从这样的顺序。而如果的确要实行清除操作,则先清除最底层的派生类,与初始化顺序刚好相反,这样做的理由时,再清除过程中派生类可能还用到基类的方法。
3.关于多态性
典型的例子如下:
public class PrivateOverride {
//private static Test monitor = new Test();
private void f(String s) {
System.out.println(”private f()”+s);
}
public static void main(String[] args) {
PrivateOverride po = new Derived();
po.f(”哈哈”);
/*monitor.expect(new String[] {
“private f()”
});*/
}
}
class Derived extends PrivateOverride {
private void f(String s) {
System.out.println(”public f()”+s);
}
} ///:~
编译不会报错,但结果不是输出:publicf()哈,因为再基类里的private方法表明它是final的,即是不会被重载的,这时侯,虽然f函数名称相同,但派生类中的f相当于以个全新的类,而基类中的f函数对派生类来说是不可见的,所以输出的为privatef)哈
再方法前加了private,static,final都可认为是早期绑定!
多态性”(Polymorphism)从另一个角度将接口从具体的实施细节中分离出来,亦即实现了“是什么”与“怎样做”两个模块的分离。利用多形性的概念,代码的组织以及可读性均能获得改善。此外,还能创建“易于扩展”的程序。无论在项目的创建过程中,还是在需要加入新特性的时候,它们都可以方便地“成长”。
3.方法调用的绑定
将一个方法调用同一个方法主体连接到一起就称为“绑定”(Binding)。若在程序运行以前执行绑定(由编译器和链接程序,如果有的话),就叫作“早期绑定”。大家以前或许从未听说过这个术语,因为它在任何程序化语言里都是不可能的。C编译器只有一种方法调用,那就是“早期绑定”。
后期绑定”,它意味着绑定在运行期间进行,以对象的类型为基础。后期绑定也叫作“动态绑定”或“运行期绑定”。若一种语言实现了后期绑定,同时必须提供一些机制,可在运行期间判断对象的类型,并分别调用适当的方法。也就是说,编译器此时依然不知道对象的类型,但方法调用机制能自己去调查,找到正确的方法主体。不同的语言对后期绑定的实现方法是有所区别的。但我们至少可以这样认为:它们都要在对象中安插某些特殊类型的信息。
Java中绑定的所有方法都采用后期绑定技术,除非一个方法已被声明成final,private,static。这意味着我们通常不必决定是否应进行后期绑定——它是自动发生的。
4.抽象类和抽象方法
Java专门提供了一种机制,名为“抽象方法”。它属于一种不完整的方法,只含有一个声明,没有方法主体。下面是抽象方法声明时采用的语法:
abstract void X();
包含了抽象方法的一个类叫作“抽象类”。如果一个类里包含了一个或多个抽象方法,类就必须指定成abstract(抽象)。否则,编译器会向我们报告一条出错消息。
若一个抽象类是不完整的,那么一旦有人试图生成那个类的一个对象,编译器又会采取什么行动呢?由于不能安全地为一个抽象类创建属于它的对象,所以会从编译器那里获得一条出错提示。通过这种方法,编译器可保证抽象类的“纯洁性”,我们不必担心会误用它。
如果从一个抽象类继承,而且想生成新类型的一个对象,就必须为基础类中的所有抽象方法提供方法定义。如果不这样做(完全可以选择不做),则衍生类也会是抽象的,而且编译器会强迫我们用abstract关键字标志那个类的“抽象”本质。
Tags: java学习 | No Comments »