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

摘要:
本文主要介绍Java中常用的子类和超类的变量实例化顺序和陷阱,并用一个Android实例讨论这个问题。出现此问题是因为Java子类和超类的实例化顺序不明确。正确的顺序如下:1.一个新的SimpleServer,其构造函数接收参数600;2.初始化父服务器的静态变量DEFAULT_PORT的赋值为900;3.要实例化子类,首先实例化其父类Server。本文参考“编写高质量代码:151条改进Java的建议”,链接1,链接2

本文主要介绍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

免责声明:文章转载自《Java拾遗(一):浅析Java子类和父类的实例化顺序 及 陷阱》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇JVM内存越多,能创建的线程越少,越容易发生java.lang.OutOfMemoryError: unable to create new native thread。JavaWeb:实现文件上传与下载下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

随便看看

极验验证码破解之selenium

大家好。我是星星在线,我又来了。今天,我给大家带来极性验证码的硒裂解方法。你有点兴奋吗?你们等不及了。让我们直奔主题。首先,随机找到一个特征点,检查元素,看它是否位于div元素,然后查看它后面的位置。距离已确定。以下是移动硒的大量模拟操作。我们只需要确认需要哪些接口。...

MySQL锁详解

MySQL锁详解update语句执行流程MySQL的锁介绍按照锁的粒度来说,MySQL主要包含三种类型(级别)的锁定机制:全局锁:锁的是整个database。由MySQL的SQLlayer层实现的表级锁:锁的是某个table。由MySQL的SQLlayer层实现的行级锁:锁的是某行数据,也可能锁定行之间的间隙。...

Cesium深入浅出之视频投影【转】

通常,我们使用矩形,因为视频形状是方形的。据怀疑,视频标签隐藏了这段关系。如果再次显示,视频将再次移动。此处使用VideoSynchronizer。它可以使视频元素与铯的模拟时钟同步。让我们看看它的构造函数:name type description optionsObject option子属性:name type默认值description用于驱动视频的...

SpringBoot工程通过Maven引入自定义Jar包

A工程为:common工程打成jar包:common-0.0.1-SNAPSHOT.jar注意:A工程打包时要使用maven的插件进行打包,不然会打成SpringBoot的Jar包,无法使用。--字符集编码--˃打包时跳过测试配置1.8˂!...

常用的Maven 插件

Maven本质上是一个插件框架。其核心不执行任何特定的施工任务。所有这些任务都委托给插件。多年来,Maven社区积累了大量经验,随后形成了成熟的插件生态系统。这个插件在一些Ant到Maven的迁移项目中特别有用。只要在父POM中配置规则,然后由每个人继承,Maven就会在规则被破坏时报告错误。...

CUPS

杯子:一个。工具1.hal设备管理器2.系统配置打印机3.Web管理器/etc/cups/ccups。conf/etc/cups/printer conf II。打印机本地安装和客户端安装1.在本地安装Linux打印机时,应选择postscript和pcl打印机。如果没有,则应将打印机设置为原始打印模式/etc/cups/printers。有限公司...