关于static的加载顺序

直接上代码1:

class Parent{
	public static void test(){
		A=3;
		System.out.println("test方法"+A);
	}
	static{
		System.out.println("par类静态块执行");
		A=2;
	}
	public static int A=1;
}

class Sub extends Parent{
	public static int B=A;
	static{
		System.out.println("sub类静态块执行");
	}
}

public class TestStaticOrder {
	public static void main(String[] args) {
		System.out.println(Sub.B);
	}
}


这段代码的执行结果是:

par类静态块执行
sub类静态块执行
1


再看代码2

class Parent{
	public static void test(){
		A=3;
		System.out.println("test方法"+A);
	}
	public static int A=1;
	static{
		System.out.println("par类静态块执行");
		A=2;
	}
}

class Sub extends Parent{
	public static int B=A;
	static{
		System.out.println("sub类静态块执行");
	}
}

public class TestStaticOrder {
	public static void main(String[] args) {
		System.out.println(Sub.B);
	}
}

这段代码的执行结果是:

par类静态块执行
sub类静态块执行
2

其实代码1和代码2, 区别只是 静态块的代码和定义静态变量代码 位置互换了一下.执行结果却不一样.


前提说明:在编译生成的class文件时,会自动产生两个方法,一个是类的初始化方法<clinit>,另一个是实例的初始化方法<init>

<clinit>方法在jvm第一次加载class文件时调用(所以静态块只会执行一次),包括静态块的执行和静态变量初始化.

<init>方法 在实例创建时候调用, 就是生成对象的时候, 例如new ,反射等等

结论:

1.虚拟机会保证在子类的<clinit>方法执行之前,父类的<clinit>方法已经执行完毕.(第一个被执行的肯定是Object类)

2.由于父类的<clinit>方法先执行,所以父类的静态块等要优先于子类的操作

3.<clinit>方法是由编译器自动收集类中的所有类变量的赋值动作和静态块中的语句合并产生的.

    编译器收集的顺序,是由语句在源文件中出现的顺序决定的.

   静态语句块中只能访问到定义在静态语句块之前的变量

  定义在它之后的变量,在前面的静态语句块中可以赋值,但是不能访问.(例如代码1中静态块直接输出A是会报错的.)

4.接口中不能使用静态语句块,单仍然有变量初始化的赋值操作,因此接口也会生成<clinit>方法.

  但是与类不同的是,执行接口的<clinit>方法不需要先执行父类接口的<clinit>方法.

  只有当父接口中定义的变量被使用时,父接口才会被初始化.

 接口的实现类在初始化时也一样不会执行接口的<clinit>方法.

5.可以看到test()方法并没有执行.static方法不在clinit的执行范围内.可以说跟加载顺序没有任何关系.




http://blog.csdn.net/methods2011/article/details/8584463 Java中的继承与静态static等的执行先后顺序的面试题 java面试题静态加载顺序构造方法 继承与static 面试题目如下:请写出程序执行完成之后的结果。 package extend; public class X { Y y=new Y(); static{ System.out.println("tttt"); } X(){ System.out.println("X"); } public static void main(String[] args) { new Z(); } } class Y{ Y(){ System.out.println("Y"); } } class Z extends X{ Y y=new Y(); static{ System.out.println("tt"); } Z(){ System.out.println("Z"); } } 先不告诉最后结果,我们先来分析下。一步一步推出结果。 1.首先分析一段程序的执行后的结果,我们得先找到程序的入口,然后才能着手分析。 也就是main()方法。 2.我们发现main()方法在X类中,要执行main()方法,还得先将X类加载到内存中。 3.X类加载完成后,会做什么事情呢?别急,先来看看static的作用,不知道吧。告诉你:static就是在类被第一次加载的时候执行,以后就不再执行。 4.知道了static的作用,那么X类被加载,那么就会先执行X类的静态属性和静态语句块(static),执行先后顺序看谁在前面就先执行谁。只在此时执行,以后都不会。 5.所以一个输出结果为tttt,没问题了吧。 6.X类的static语句块执行完了,就该执行main()方法啦。 7.new Z();此方法被执行。 8.既然new Z();那么Z类就要被加载。因为Z类继承X类。所以必须先加载X类才行。因为X类已经被加载。所以此时不用再加载X类了。Z类加载好了就要执行Z类的static语句块 9.那么就会打印出tt了吧。 10.都加在完后就要实例化对象了。 11.实例化Z之前,还得先实例化X对吧。因为子类的构造方法都会调用父类的构造方法。 12.那就先实例化X类吧。 13.执行X方法前还得先初始化对不。也就是获取所有属性。那么X类的属性Y就会获取。 14.即X类的Y y=new Y();要被执行。也就是会打印Y。 15.接着执行System.out.println("X"); 16.然后就是执行Z的构造方法 17.同样先获取Z的属性Y y=new Y();打印Y。 18.再执行System.out.println("Z"); 整个过程就是这样了。现在知道结果了吧: tttt tt Y X Y Z http://snow4909.diandian.com/post/2013-02-17/40049419937 记住,面试有可能会标出一些语句的,让你选择这些语句的执行顺序。不过只要知道原理,就没什么难的了。 执行先后顺序: 1、类加载到内存时static 加载进内润 2、调用类的构造方法时先调用父类的构造方法,在调用子类的构造方法 3、类初始化时,先初始化类的属性成员,在执行构造方法。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值