java 编程思想之7.2.1初始化基类
2015-08-27 10:54
246 查看
继承是所有OOP语言不可缺少的能成部分。但继承并不只是复制基类的接口。当创建一个导出类的对象时,该对象包含了一个基类的子对象。这个子对象与你用基类直接创建的对象是一样的。二者的区别在于,后者来自于外部,而基类的子对象被包装在导出类对象内部。
对基类子对象的正确初始化顺序之前不能说不知道,好像所有知道点都一样,只是去看,就觉得自己掌握了,其实不是。那只是当时的自以为。呵呵,眼高手低。今天看到相关的章节,翻翻又是觉得都是自己知道的呀。也许是对同一个知识点有过太多次这种想,于是今天决定不只是看,不只是自以为的知道。动动手敲些例子吧。做了开发三年半了,也不算是个菜鸟了吧。之前项目一直安排相当紧凑,三年多大多只是在实现功能,赶进度。少知又少的去研究本质。现在处于项目空闲期也三四个月了。一度觉得应该回归本质好好书本及语言本质而不是一味的实现实现。好吧,开了个小差,回归正题。先看我敲的简单不能基础了的JAVA小示例:
对基类子对象的正确初始化,仅有一种方法可以保证这一点,在构造器中调用基类构造器来执行初始化,而基类构造器具有执行基类初始化所需要的所有知识能力。Java会自动在导出类的构造器中插入对基类构造器的调用。
默认构造器
package com.ebao.java.constructor;
public class ClassA {
ClassA(){
System.out.println("==============ClassA");
}
}
package com.ebao.java.constructor;
public class ClassB {
ClassB(){
System.out.println("==============ClassB");
}
}
package com.ebao.java.constructor;
public class ClassC extends ClassA {
ClassB cb = new ClassB();
public static void main(String[] arg){
ClassC cc = new ClassC();
}
}
运行结果:
//==============ClassA
//==============ClassB
由此看出实例一个类时,构建过程是从基类向下扩展。然后是成员对象。所以基类构造器总是会被调用且是在导出类构造器之前被调用。
带参数的构造器
上面的例子中都是默认构造器,也就是都是不带参数的构造器。编译器可以轻松的调用它们是因为不必考虑要传递什么样的参数的问题。但是如果没有默认的基类构造器,或者想调用一个带参数的基类构造器。就必须用super显式地编写调用基类构造器的语句,并配以适当的参数列表:
public class Game {
Game(int i){
System.out.println("---------------------Game");
}
}
package com.ebao.java.constructor;
public class BoardGame extends Game {
BoardGame(int i){
super(i);
System.out.println("--------------------BoardGame");
}
}
package com.ebao.java.constructor;
public class Chess extends BoardGame {
Chess(int i) {
super(i);
System.out.println("--------------------Chess");
}
public static void main(String[] arg){
Chess cs = new Chess(7);
}
}
运行结果:
---------------------Game
--------------------BoardGame
--------------------Chess
如果不在子类中显示调用基类构造器或调用类基构造器不是第一句编译报错如下
对基类子对象的正确初始化顺序之前不能说不知道,好像所有知道点都一样,只是去看,就觉得自己掌握了,其实不是。那只是当时的自以为。呵呵,眼高手低。今天看到相关的章节,翻翻又是觉得都是自己知道的呀。也许是对同一个知识点有过太多次这种想,于是今天决定不只是看,不只是自以为的知道。动动手敲些例子吧。做了开发三年半了,也不算是个菜鸟了吧。之前项目一直安排相当紧凑,三年多大多只是在实现功能,赶进度。少知又少的去研究本质。现在处于项目空闲期也三四个月了。一度觉得应该回归本质好好书本及语言本质而不是一味的实现实现。好吧,开了个小差,回归正题。先看我敲的简单不能基础了的JAVA小示例:
对基类子对象的正确初始化,仅有一种方法可以保证这一点,在构造器中调用基类构造器来执行初始化,而基类构造器具有执行基类初始化所需要的所有知识能力。Java会自动在导出类的构造器中插入对基类构造器的调用。
默认构造器
package com.ebao.java.constructor;
public class ClassA {
ClassA(){
System.out.println("==============ClassA");
}
}
package com.ebao.java.constructor;
public class ClassB {
ClassB(){
System.out.println("==============ClassB");
}
}
package com.ebao.java.constructor;
public class ClassC extends ClassA {
ClassB cb = new ClassB();
public static void main(String[] arg){
ClassC cc = new ClassC();
}
}
运行结果:
//==============ClassA
//==============ClassB
由此看出实例一个类时,构建过程是从基类向下扩展。然后是成员对象。所以基类构造器总是会被调用且是在导出类构造器之前被调用。
带参数的构造器
上面的例子中都是默认构造器,也就是都是不带参数的构造器。编译器可以轻松的调用它们是因为不必考虑要传递什么样的参数的问题。但是如果没有默认的基类构造器,或者想调用一个带参数的基类构造器。就必须用super显式地编写调用基类构造器的语句,并配以适当的参数列表:
public class Game {
Game(int i){
System.out.println("---------------------Game");
}
}
package com.ebao.java.constructor;
public class BoardGame extends Game {
BoardGame(int i){
super(i);
System.out.println("--------------------BoardGame");
}
}
package com.ebao.java.constructor;
public class Chess extends BoardGame {
Chess(int i) {
super(i);
System.out.println("--------------------Chess");
}
public static void main(String[] arg){
Chess cs = new Chess(7);
}
}
运行结果:
---------------------Game
--------------------BoardGame
--------------------Chess
如果不在子类中显示调用基类构造器或调用类基构造器不是第一句编译报错如下
相关文章推荐
- Java判断浏览器类型
- Java基础知识强化01:short s = 1; s = s + 1;与short s = 1; s += 1;
- 个人环境搭建——搭建JDK环境
- HBase之Java API实操
- 基于restful风格的maven项目实践(融合spring)
- Quartz集成Spring框架
- 关于javax.servlet不存在的问题
- leetcode--PalindromeLinkedList
- javaweb学习总结(二十八)——JSTL标签库之核心标签
- JAVA设置session超时失效的方式
- Struts 2读书笔记-----使用Struts 2的输入校验
- Java 类加载机制 ClassLoader Class.forName 内存管理 垃圾回收GC
- Java 类加载机制 ClassLoader Class.forName 内存管理 垃圾回收GC
- java WEB应用程序启动时是如何加载AXIS server-config.wsdd 的
- java - properties read write
- springMVC框架的搭建最简单项目(没有数据库的配置)
- [转自wangpeng047]Eclipse使用入门教程
- java - mysql连接
- Java常用集合类
- javax.naming.NameNotFoundException