Java中为了让多个对象拥有同一个属性,可以通过static关键字将某一个共享的属性设置为全局变量,static变量存储在JVM中的data segment中,所有线程共享.那么,带有static关键字的属性和非static属性的初始化顺序又是怎样的呢?本篇文章中将附带一道关于该类问题的笔试题.
一.static介绍
在JVM中,对象本身被存放在heap segment(堆)中,对象的引用存放在stack segment(栈)中,被static修饰的属性存放在方法区中,作为一个全局可以被所有线程调用.被static修饰的属性、方法或初始化区域,都只会在类加载的时候被执行,并且存放在方法区中可以被所有线程所共享.
二.笔试题练习
public class Test1 {
public static int k = 0;
public static Test1 t1 = new Test1("t1");
public static Test1 t2 = new Test1("t2");
public static int i = print("i");
public static int n = 99;
public int j = print("j");
{
print("构造块");
}
static{
print("静态块");
}
public Test1(String str){
System.out.println((++k)+":"+str+" i="+i+" n="+n);
++i;++n;
}
public static int print(String str){
System.out.println((++k)+":"+str+" i="+i+" n="+n);
++n;
return ++i;
}
public static void main(String[] args) {
Test1 t = new Test1("init");
}
}
执行结果如下所示:
现在我们来分析一下:
在主程序中创建Test1对象时会先加载Test1这个类,执行该类中的所有静态成员变量(实例域)和static块语句.在这一过程中程序是顺序执行的.当程序执行到public static Test1 t1 = new Test1("t1")时,程序会卡在当前位置转而去创建这个新对象t1,这一过程类似于递归的执行.而在执行new Test1("t1")时,因为已经加载过该类的所有静态成员,所以所有的静态区域都不再执行,此时只会执行非static区域.那么程序输出的第一至第三行就能讲的通了.该过程流程如下:
值得注意的是,静态成员开始执行不是在new对象的时候执行,而是在Test1 t;的时候执行.
那么问题来了,为什么输出的n的值不是static中定义好的99而是0呢???
其实这是因为对于static属性来说,类在加载的一开始就会将所有的属性名存在方法区中,此时相当于只声明了这些属性,但是没有进行赋值,所以此时他们都有各自的默认值,对于t1和t2来说他们的默认值为null,而int的默认值是0.由于赋值语句的顺序执行,只有Test1 t1被赋值后才会执行后面的语句,所以对n的赋值要在t1、t2、i都被赋值之后,才能给n赋值.
剩下的几行大家可以自行推理判断,实在不行代码拿走改一改,自己进行推算..