Thinking In Java 4th Chap7 复用类

复用代码的两种方法:组合和继承

组合方法:(新类中产生现有类的对象)

  没什么好说的,就是调用别类的对象而已

  值得一提的是一个特殊方法:toString()当需要一个String却只有对象时能够自动调用此方法(每个类只需写一个toString方法)

    例如:class Myclass{

        /*..............*/

        private String s;

        public String toString(){

          return s;} }

  对象引用的初始化方法:1.定义对象之时即初始化  2.在类的构造器中初始化  3.在使用对象之前才初始化  4.实例初始化

  

 1 class Soap {
 2     public Soap() {
 3         System.out.println("Soap");
 4     }
 5 
 6     @Override
 7     public String toString() {
 8         return "Constructed";
 9     }
10 }
11 
12 public class TestInit {
13     private String s1 = "Happy"; // 定义初始化
14     private Soap s2;
15     private String s3, s4;
16 
17     // 实例初始化
18     {
19         s2 = new Soap();
20     }
21 
22     public TestInit() {
23         s3 = "Good"; // 构造器初始化
24     }
25 
26     @Override
27     public String toString() {
28         s4 = "Girl"; // 惰性初始化
29         return "TestInit{" + "s1='" + s1 + ", s2=" + s2 +
30                 ", s3='" + s3 + ", s4='" + s4 + '}';
31     }
32 
33     public static void main(String[] args) {
34         TestInit ti = new TestInit();
35         System.out.println(ti);
36     }
37 }
38 /*
39     Output:
40     Soap
41     TestInit{s1='Happy, s2=Constructed, s3='Good, s4='Girl}
42  */

继承方法:(按照现有类的类型创建新类)

  除非指明要继承的类,否则隐式地从根类Object进行继承

  例如:public class Detergent extends Cleaner{/*...................*/}类Detergent继承自类Cleaner,获得Cleaner的全部域和方法

  通常为了继承,将所有基类的数据成员定义为private,将所有方法定义为public(或protected)

  想要调用基类的方法,使用super关键字super.funcFromBase();

  调用继承类对象的同时不仅调用了自己的构造器,也调用了基类的构造器(而且是在调用自己的构造器之前)

    如果基类的构造器带有参数,调用基类的构造器需要使用super:

    例如:class Base{

        Base(int i){/*...............*/} }

       class Myclass extends Base{

        Myclass(int i){

        super(i);

        /*...................*/} }

介乎组合和继承之间还存在一种方法:代理(在自己的类中建立别类的对象,在自己的方法中调用对象的方法)

  例如://在类的域中

     Baseclass bc = new Baseclass();

     void Myfunc(){

      bc.Myfunc();}

     /*.............*/

  使用代理的好处是我们可以控制使用“基类”的部分成员而不用全盘“拿来主义”

其中一种清理方式是try-finally语句组合(即使需要手动清理,也不要使用finalize())

  try{

  //需要被特殊处理的代码}

  finally{

  //不管怎样,只要try语句块内代码执行完成就会调用的代码}

不同于c++:继承类重新定义基类的某个多次重载的方法,不会屏蔽该方法在基类中的任何一个版本

@override注解:想要覆写某个方法却不小心重载了该方法时,编译器会生成一条错误信息

覆写:方法名和参数列表都完全一致  重载:方法名一致但是参数列表不同

如果是“是一个”的关系,应当使用继承(实际上慎用继承是明智的选择,如果不是必须向上转型,那么继承则不是必要的);

如果是“有一个”的关系,应当使用组合(组合在一般情况下在新类中应使调用对象域为private)

向上转型

  其中一例是在继承类中的对象作为参数可以使用基类中的方法(虽然基类中的方法的参数并非继承类中的对象类型,却可以使用)

  例如:class Base{

      play(Base bs){/*.................*/} }

     public class Son extends Base{

      Son sn=new Son();

      Base.play(sn);}这里将子类引用转换为基类引用

final关键字:(类比c++中的const关键字)

  编译期常量,这类常量必须是基本数据类型而且以final修饰

  对象引用可以是常量,也就是说永远只指向同一对象(但是指向的对象的值是可以改变的,Java并不提供任何使对象恒定不变的途径)

  同时以static和final修饰的只占有一份不变空间的常量,其名称全部大写并以_分隔单词

  使用final参数意味着无法在方法中改变参数引用所指向的对象,也无法改变基本数据的值:

    void func(final Myclass mc){

      //! mc=new Myclass(); mc因为是final修饰的,所以无权再指向新的对象

      /*...............*/}

  final方法:将方法锁定,防止继承类修改其含义

    类中所有的private方法都隐式的认为是final,如果在继承类中写一个同名的方法(并无向上引用的接口),仅仅是同名而已(与基类方法并无关联)

    只有方法是基类的接口的一部分(即能够将一个对象向上引用为基类的基本类型并调用相同方法),才能体现“覆盖”效应(无private或final的保护)

  final类:如果将类整体定义为final(例如:final class Myclass{/*..........*/}),则断绝了继承该类的途径,而且类中所有的方法都隐式指定为final

所有static对象和代码段按定义类时的书写顺序而依次初始化

类是在其任何static成员被访问时加载的(构造器是隐式的static成员)

调用一个继承类的初始化顺序:基类->(可能存在的其他基类)->继承类

 

  

 

上一篇:用JS编写一个函数,返回数组中重复出现过的元素


下一篇:Design Thinking: Understanding the Process 设计思维:理解过程 Lynda课程中文字幕