Java的成员初始化方式包括:默认初始化(自动初始化)、指定初始化、构造器初始化、静态块初始化(静态子句)、实例初始化(非静态实例初始化)
1.默认初始化:即只进行成员变量的定义,不进行主动的初始化操作,由编译器为成员变量赋一个默认值。
public class InitialValue{
boolean bool;
char ch;
byte b;
short s;
int i;
long lng;
float f;
double d;
Depth depth;
}
class Depth{}
对于基本类型,有各自不同的默认值如int、char、short等为0,boolean为false,对于对象,默认值为null。默认初始化主要是为了保证初始化过程一定会得到执行,只发生在成员变量身上,对于局部变量,不存在这种机制。
2.指定初始化:即在成员变量定义处给出变量的初值
public class InitialValue{
boolean bool = true;
char ch = 'x';
byte b = 47;
short s = 0xff;
int i = 99;
long lng = 1;
float f = 3.14f;
double d =3.14159;
Depth depth = new Depty();
}
class Depth{}
这种初始化方式既直观又简单,但是类InitialValue的所有对象都会具有相同的初值。
3.构造器初始化:即使用构造器方法来进行初始化,既可以使用默认构造器,也可自己创建构造器。
public class InitialValue{
boolean bool;
char ch;
byte b;
Depth depth;
public InitialValue(boolean bool,char ch,byte b,Depty depty){
this.bool = bool;
this.ch = ch;
this.b = b;
this.depth = depth;
}
}
class Depth{}
构造器不会阻止自动初始化的进行,且自动初始化会在构造器被调用之前进行。例如上述代码中的bool会首先自动初始化为false,然后才会调用构造器初始化为参数bool的值。虽然很多情况下会造成二次初始化的问题,但这种机制保证了初始化一定得到了执行。
当我们在一个类中不创建任何构造函数时,类会存在一个无参且不进行任何操作的默认构造函数,但是一旦我们创建了自己的构造函数,默认构造函数就会被覆盖,再想要调用,就需要自己显式的定义出来。
4.静态块初始化:是一种显示的静态数据初始化方式,将多个静态初始化动作组织成一个特殊的“静态子句”。
在讨论静态块初始化前,先讨论一下静态数据初始化。对于静态数据来说,默认初始化和指定初始化与非静态数据是没有什么分别的,但是由于无论创建多少对象,静态数据都只占用一份存储区域,对于同一个类的所有对象,静态数据的值是相同的,也就是说,无论创建多少类的对象,静态数据只会初始化一次,因此用构造器方式初始化是不合适的。
静态初始化只会在必要的时刻进行,即第一个该类的对象被创建,或者第一次访问静态数据(静态数据在创建类的对象就可以访问)。此后,静态对象不会再次被初始化。
public class IntialValue{
static int i;
static {
i = 47;
}
}
第二个static及花括号中的内容就是静态子句,这段代码与其他类型的初始化动作执行时机是相同的。花括号中也可以包括非赋值操作的初始化操作。
5.非静态实例初始化:与静态子句类似,但是是用于非静态变量的初始化。
public class Mugs{
Mug mug1;
Mug mug2;
{
mug1 = new Mug(1);
mug2 = new Mug(2);
print("mug1 & mug2 initialized");
}
}
这种语法对于支持“匿名内部类”的初始化是必须的,但是它也使得你无论调用了哪个构造器,某些操作都会发生。实例初始化同样发生在构造器调用之前。
在讨论完Java的初始化方式以后,我们来讨论一下Java的初始化顺序:
1.指定初始化,实例初始化,静态子句都发生在构造器被调用之前。
2.静态初始化会先于非静态初始化,且静态初始化只会进行一次。
3.无论何时,默认初始化都会执行,这是因为在堆上为对象分配存储空间的同时,会自动将该空间清零,也就进行了默认初始化。
参考文献:
1.Bruce, Eckel. Java编程思想(第四版)[M]. 机械工业出版社:华章公司, 2007. 91-98