当类型属于被动使用的时候,它将不会被初始化。
请看下面的例子:
package cantellow.text11;
public class Example1
{
/**
* @param args
* @author cantellow
*/
public static void main(String[] args)
{
// TODO Auto-generated method stub
int hours = NewbornBaby.hoursOfSleep;
System.out.println(hours);
}
static
{
System.out.println("Example1 was initialized.");
}
}
class NewParent
{
static int hoursOfSleep = (int) (Math.random()*3.0);
static
{
System.out.println("NewParent was initialized.");
}
}
class NewbornBaby extends NewParent
{
static int hourOfCrying = 6 + (int)(Math.random()*2.0);
static
{
System.out.println("NewbornBaby was initialized.");
}
}
结果是:
Example1 was initialized.
NewParent was initialized.
2
问题:为什么 NewbornBaby类没有被初始化?
=======================================================
参考理解:
接口中的声明的字段可能会被子接口或者实现了这个接口的类引用。对于子类、字接口和实现了接口的类来说,这就是被动使用。所以,在没有明确指定使用NewbornBaby类的hourOfCrying 字段时,NewbornBaby类是不会初始化的。
下面这六种情形符合主动使用的要求:
1 、当创建某个类的新实例时(或者通过在字节码中执行new指令;或者通过不明确的创建、反射、克隆或者反序列化)。
2、当调用某的类的静态方法时(即在字节码中执行invokestatic指令时)。
3、当使用某个类或接口的静态字段,或者对该字段赋值时(即在字节码中,执行getstatic或putstatic指令时),用final修饰的静态字段除外,它被初始化为一个编译时的常量表达式。
4、当调用api中的某些反射方法时,比如类class中的方法或者java.lang.reflect包中的类方法。
5、当初始化某个类的子类时,(某个类初始化时,要求它的超类已经被初始化了)
6、当虚拟机启动某个被表明为启动类的类(即含有main方法的那个类)