概要:本文通过查看一个精心构造的类结构的运行输出和使用Javap工具查看实际生成的java字节码(bytecode)向java程序员展示了一个类在运行时是如何构造生成的。
要害字: java 构造 javap 字节码 bytecode
按照java规范,一个类实例的构造过程是遵循以下顺序的:
1.假如构造方法(constrUCtor,也有翻译为构造器和构造函数的)是有参数的则进行参数绑定。
2.内存分配将非静态成员赋予初始值(原始类型的成员的值为规定值,例如int型为0,float型为0.0f,boolean型为false;对象类型的初始值为null),静态成员是属于类对象而非类实例,所以类实例的生成不进行静态成员的构造或者初始化,后面将讲述静态成员的生成时间。
3.假如构造方法中存在this()调用(可以是其它带参数的this()调用)则执行之,执行完毕后进入第6步继续执行,假如没有this调用则进行下一步。
4.执行显式的super()调用(可以是其它带参数的super()调用)或者隐式的super()调用(缺省构造方法),此步骤又进入一个父类的构造过程并一直上推至Object对象的构造。
5.执行类申明中的成员赋值和初始化块。
6.执行构造方法中的其它语句。
现在来看看精心构造的一个实例:
class Parent
{
int pm1;
int pm2=10;
int pm3=pmethod();
{
System.out.println("Parent's instance initialize block");
}
public static int spm1=10;
static
{
System.out.println("Parent's static initialize block");
}
Parent()
{
System.out.println("Parent's default constructor");
}
static void staticmethod()
{
System.out.println("Parent's staticmethod");
}
int pmethod()
{
System.out.println("Parent's method");
return 3;
}
}
class Child extends Parent
{
int cm1;
int cm2=10;
int cm3=cmethod();
Other co;
public static int scm1=10;
{
System.out.println("Child's instance initialize block");
}
static
{
System.out.println("Child's static initialize block");
}
Child()
{
co=new Other();
System.out.println("Child's default constructor");
}
Child(int m)
{
this();
cm1=m;
System.out.println("Child's self-define constructor");
}
static void staticmethod()
{
System.out.println("Child's staticmethod");
}
int cmethod()
{
System.out.println("Child's method");
return 3;
}
}
class Other
{
int om1;
Other() {
System.out.println("Other's default constructor");
}
}
public class InitializationTest
{
public static void main(String args[])
{
Child c;
System.out.println("program start");
System.out.println(Child.scm1);