Java拾遗(一):浅析Java子类和父类的实例化顺序 及 陷阱

本文主要介绍Java里经常使用的子类和父类的变量实例化顺序及陷阱,并结合一个Android实例来探讨此问题。日后编程中应尽量避免此陷阱。

首先看以下一段代码:

定义一个虚类Server.java

package org.yanzi.test1;

public abstract class Server {
	private static final int DEFAULT_PORT = 900;
	public Server() {
		// TODO Auto-generated constructor stub
		int port = getPort();
		System.out.println("port = " + port + " DEFAULT_PORT = " + DEFAULT_PORT);
	}
	protected abstract int getPort();

}

然后定义一个子类SimpleServer.java

package org.yanzi.test1;

public class SimpleServer extends Server {
private int mPort = 100;
	public SimpleServer(int port) {
		// TODO Auto-generated constructor stub
		this.mPort = port;
	}
	@Override
	protected int getPort() {
		// TODO Auto-generated method stub
		return mPort;
	}

}

測试代码:

package org.yanzi.test1;

public class Test1 {

	/**
	 * @param args
	 */
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Server s = new SimpleServer(600);
	}

}

測试结果:

port = 0 DEFAULT_PORT = 900

在測试代码里,传了一个參数600,我们希望getPort得到也是600,但遗憾的是得到的是一个大大的0!!!出现这个问题是因对Java子类和父类实例化顺序存在模糊,以下来看下其正确顺序:

1、new一个SimpleServer,SimpleServer的构造函数接收參数600;

2、初始化父类Server的静态变量,DEFAULT_PORT赋值为900;

3、为了实例化子类,首先实例化其父类Server。子类有參数的构造中默认包含了super方法,即调用父类的无參构造函数。因此就到了int port = getPort();这一句,调用子类的getPort方法。子类的getPort方法返回mPort,此时mPort还没有赋值,因此还是0.这就是得到大大的0的原因!!!

4、父类初始化完成后,開始初始化子类的实例变量,mPort赋值100;

5、运行子类的构造函数,mPort赋值600;

6、子类实例化完成,对象创建完了!

真相大白了,再做一个測试。将SimpleServer里的实例变量mPort搞成静态变量例如以下:

package org.yanzi.test1;

public class SimpleServer extends Server {
private static int mPort = 100;
	public SimpleServer(int port) {
		// TODO Auto-generated constructor stub
		this.mPort = port;
	}
	@Override
	protected int getPort() {
		// TODO Auto-generated method stub
		return mPort;
	}

}

測试结果:

port = 100 DEFAULT_PORT = 900

其运行顺序是:

1.第一个步骤同上,SimpleServer接收构造參数600

2.初始化父类的静态代码块,当然包含静态变量,然后初始化子类的静态变量

3.初始化父类的非静态代码,包含非静态的变量等;

4.运行父类的构造函数;

5.初始化子类的非静态代码

6.运行子类的构造函数。

至此完成,终于的结论就是构造函数越简单越好,不要在构造函数里做太多操作。回过头再来看杂家的前文:Android自己定义UI陷阱:LayoutInflater.from().inflate()一定不能工作在父类或虚类里 假设把initView()放在父类里,则子类LAYOUT_ID在使用时还会是0.因此即便要用也要将此弄成static类型的。

本文參考《编写高质量代码:改善Java的151个建议》、链接1链接2

时间: 2024-10-09 17:36:28

Java拾遗(一):浅析Java子类和父类的实例化顺序 及 陷阱的相关文章

java中子类继承父类程序执行顺序问题

Java中,new一个类的对象,类里面的静态代码块.非静态代码.无参构造方法.有参构造方法.类的一般方法等部分,它们的执行顺序相对来说比较简单,用程序也很容易验证.比如新建一个测试父类. public class FatherTest { private String name; FatherTest(){ System.out.println("--父类的无参构造函数--"); } FatherTest(String name){ this.name=name; System.out

Java入门-浅析Java学习从入门到精通【转】

一. JDK (Java Development Kit)  JDK是整个Java的核心,包括了Java运行环境(Java Runtime Envirnment),一堆Java工具和Java基础的类库(rt.jar).不论什幺Java应用服务器实质都是内置了某个版本的JDK.因此掌握 JDK是学好Java的第一步.最主流的JDK是Sun公司发布的JDK,除了Sun之外,还有很多公司和组织都开发了自己的JDK,例如IBM公司开发 的JDK,BEA公司的Jrocket,还有GNU组织开发的JDK等等

Java 中变量初始化、子类和父类构造器调用的顺序

先说结论 变量初始化 -> 父类构造器 -> 子类构造器 贴代码 Animcal.java 父类 public class Animal { private static int index = 0; private static String getStaticName() { String name = String.format("animal %d" , index++); System.out.println(name); return name; } priva

JAVA面试题 浅析Java中的static关键字

面试官Q1:请说说static关键字,你在项目中是怎么使用的? static 关键字可以用来修饰:属性.方法.内部类.代码块: static 修饰的资源属于类级别,是全体对象实例共享的资源: 使用 static 修饰的属性,静态属性是在类的加载期间初始化的,使用类名.属性访问 案例说明 ①修饰成员变量 package com.ant.param; public class StaticFieldDemo { public static void main(String[] args) { Foo

虚拟继承中子类和父类的构造函数顺序1

这里的inter1,2,3.要写base2的构造函数 final也写,否则里面的数据未初始化 inert2写的原因是假如有人使用直接使用inert2,里面数据就可以初始化了 总结一下,虚拟继承的基类就像一个指针一样差不多,但又有分别,因为弄了 private会使用不了数据. 以后每个要实现的类都要调用虚拟基类的构造函数,如上面的inter2. 写上我个人测试代码 class A{ protected://不能写private int a; public: A(int x) :a(x){} };

Java中子类和父类的构造函数?

这篇文章总结了关于Java构造的常见??问题. 1)为什么创建一个子类对象要也需要调用父类的构造函数? class Super { String s; public Super(){ System.out.println("Super"); } } public class Sub extends Super { public Sub(){ System.out.println("Sub"); } public static void main(String[] a

java 子类重写父类的方法应注意的问题

若想实现一个合格重写方法,而不是重载,那么必须同时满足下面的要求! A.重写规则之一:    重写方法不能比被重写方法限制有更严格的访问级别.(但是可以更广泛,比如父类方法是包访问权限,子类的重写方法是public访问权限.) 比如:Object类有个toString()方法,开始重写这个方法的时候我们总容易忘记public修饰符,编译器当然不会放过任何教训我们 的机会.出错的原因就是:没有加任何访问修饰符的方法具有包访问权限,包访问权限比public当然要严格了,所以编译器会报错的. 反正子类

[转]Java中子类调用父类构造方法的问题分析

在Java中,子类的构造过程中,必须调用其父类的构造函数,是因为有继承关系存在时,子类要把父类的内容继承下来,通过什么手段做到的? 答案如下:    当你new一个子类对象的时候,必须首先要new一个父类的对像出来,这个父类对象位于子类对象的内部,所以说,子类对象比父类对象大,子类对象里面包含了一个父类的对象,这是内存中真实的情况.构造方法是new一个对象的时候,必须要调的方法,这是规定,要new父类对象出来,那么肯定要调用其构造方法,所以: 第一个规则:子类的构造过程中,必须调用其父类的构造方

java子类与父类的关系

摘自翁恺老师的mooc 对理解继承来说,最重要的事情是,知道哪些东西被继承了,或者说,子类从父类那里得到了什么.答案是:所有的东西,所有的父类的成员,包括变量和方法,都成为了子类的成员,除了构造方法.构造方法是父类所独有的,因为它们的名字就是类的名字,所以父类的构造方法在子类中不存在.除此之外,子类继承得到了父类所有的成员. 但是得到不等于可以随便使用.每个成员有不同的访问属性,子类继承得到了父类所有的成员,但是不同的访问属性使得子类在使用这些成员时有所不同:有些父类的成员直接成为子类的对外的界