复用代码
复用性是面向对象技术带来的很棒的潜在好处之一。如果运用的好的话可以帮助我们节省很多开发时间,提升开发效率。但是,如果被滥用那么就可能产生很多难以维护的代码。
Java代码的复用有继承,组合以及代理三种具体的表现形式。
- 继承:继承(Inheritance)是一种联结类与类的层次模型。指的是一个类(称为子类、子接口)继承另外的一个类(称为父类、父接口)的功能,并可以增加它自己的新功能的能力,继承是类与类或者接口与接口之间最常见的关系;继承是一种
is-a
关系。 - 组合:组合(Composition)体现的是整体与部分、拥有的关系,即
has-a
的关系。
组合
在新类中创建已有类的对象,通过该对象来调用已有类中的非private的属性和方法
基本类型初始化0。因为Java不会帮我们自动创建一个缺省的对象,它只会自动替我们把字段中的引用初始化为null。
为引用赋值可以在三个地方
-
在定义这个引用的时候
-
在构造函数中,
-
即将要使用这个对象之前
-
使用实例初始化
继承
使用extends
关键字在基类的基础上创建新类,新类可以直接复用基类的非private
的属性和方法
- 子类和基类对象创建的顺序是怎么样的呢?
为了能够正确的初始化基类,一般会调用基类的构造函数来进行初始化。Java中在调用子类的构造函数时首先会自动的调用基类的构造函数,并且这样的过程是层层传递的。
但是这里自动调用的都是基类的默认构造函数(无参的),如果我们想调用基类的某个带参数的构造函数又该怎么办呢?上面提到可以用super来代替基类的引用,与在构造函数中通过this调用本类其它构造函数的形式一样,我们可以通过super来调用基类带参数的构造函数,比如“super(i, j)”。
代理
在新类中创建代理,通过代理来操作已有类的非private的属性和方法
final
作为代码的生产者来说,我们有时候希望代码被复用,有的时候又希望代码不被复用。当我们不想代码被复用时,final关键字就派上用场了。final这个关键字很形象,它本身就说明了一切——最后的,最终的;决定性的;不可更改的。
使用final的场景有三种,分别是数据、方法和类。我们来稍作说明。
1)final 数据
- 最常见的final数据就是常量了,例如:
public class Consts {
public static final String CMOWER = "沉默王二";
}
对于常量来说,它对于整个应用内的所有类都是可见的,因此是public的;它可以直接通过类名.常量名访问,所以是static的;它是不可修改的,因此是final的。
- 空白final
被申明为final又为初始化的域,在使用之前必须被初始化。
- 另外一种常见的final数据就是参数了,参照程序清单4-1。
程序清单4-1:
public class Cmower {
public void write(final String content) {
// content += "犹未雪"; // final修饰的参数是无法在方法内部被再次修改的
System.out.println(content);
}
public void write1(String content) {
content += "犹未雪";
System.out.println(content);
}
public static void main(String[] args) {
Cmower cmower = new Cmower();
cmower.write("精忠报国");
cmower.write1("靖康耻");
}
}
2)final 方法
在Java类中,所有的private方法都隐式地指定为final的(也就是说,如果你在private方法上加上final修饰符,其实是没啥意义的)。在介绍继承的时候,你应该注意到我强调的一句话,就是新类可以直接复用基类的非private的属性和方法,也就是说private方法是无法被继承者修改的,因为private方法是final的。
来看程序清单4-2,你会发现Wangsan类型的san引用是不能调用say(String words)方法的,因为private方法是无法被继承者修改的,尽管Wangxiaosan中重新定义了say(String words)方法。
程序清单4-2:
public class Wangxiaosan extends Wangsan {
public Wangxiaosan() {
say("吃中饭没");
}
public void say(String words) {
System.out.println("王小三在说:" + words);
}
public static void main(String[] args) {
Wangsan san = new Wangxiaosan();
// san.say("吃晚餐没"); // 无法访问,并不会被覆盖
}
}
class Wangsan {
public Wangsan() {
say("吃早饭没");
}
private void say(String words) {
System.out.println("王三在说:" + words);
}
}
3)final 类
当我们认为某个类就是最终的形态了,它很完美,不应该被继承,就可以使用final关键字来修饰;参照程序清单4-3。
程序清单4-3:
// 无法继承
public class Wangxiaosan extends Wangsan {
}
final class Wangsan {
public Wangsan() {
System.out.println("我就是最终形态,别继承我!");
}
}