java什么时候会触发类加载_java中类的加载,及执行顺序

类的加载顺序

什么时候类加载:第一次需要使用类信息时加载。

类加载的原则:延迟加载,能不加载就不加载。

触发类加载的几种情况:

1、调用静态成员时,会加载静态成员真正所在的类及其父类。 通过子类调用父类的静态成员时,只会加载父类而不会加载子类。

2、第一次 new 对象的时候 加载(第二次再 new 同一个类时,不需再加载)。

3、加载子类会先加载父类。(覆盖父类方法时所抛出的异常不能超过父类定义的范围)

注:如果静态属性有 final 修饰时,则不会加载,当成常量使用。

例:public static final int a =123;

但是如果上面的等式右值改成表达式(且该表达式在编译时不能确定其值)时则会加载类。

例:public static final int a = math.PI

如果访问的是类的公开静态常量,那么如果编译器在编译的时候能确定这个常量的值,就不会被加载;

如果编译时不能确定其值的话,则运行时加载

类加载的顺序:

1、加载静态成员/代码块: 先递归地加载父类的静态成员/代码块(Object的最先);再依次加载到本类的静态成员。

同一个类里的静态成员/代码块,按写代码的顺序加载。

如果其间调用静态方法,则调用时会先运行静态方法,再继续加载。同一个类里调用静态方法时,可以不理会写代码的顺序。

调用父类的静态成员,可以像调用自己的一样;但调用其子类的静态成员,必须使用“子类名.成员名”来调用。

2、加载非静态成员/代码块:(实例块在创建对象时才会被加载。而静态成员在不创建对象时可以加载)

先递归地加载父类的非静态成员/代码块(Object的最先);再依次加载到本类的非静态成员。

同一个类里的非静态成员/代码块,按写代码的顺序加载。同一个类里调用方法时,可以不理会写代码的顺序。

但调用属性时,必须注意加载顺序。一般编译不通过,如果能在加载前调用,值为默认初始值(如:null 或者 0)。 调用父类的非静态成员(private

除外),也可以像调用自己的一样。

3、调用构造方法:先递归地调用父类的构造方法(Object的最先)也就是上溯下行;默认调用父类空参的,也可在第一行写明调用父类某个带参的。再依次到本类的构造方法;构造方法内,也可在第一行写明调用某个本类其它的构造方法。 注意:如果加载时遇到 override

的成员,可看作是所需创建的类型赋值给当前类型。

其调用按多态用法:只有非静态方法有多态;而静态方法、静态属性、非静态属性都没有多态。

假设子类override父类的所有成员,包括静态成员、非静态属性和非静态方法。

由于构造子类时会先构造父类;而构造父类时,其所用的静态成员和非静态属性是父类的,但非静态方法却是子类的;

由于构造父类时,子类并未加载;如果此时所调用的非静态方法里有成员,则这个成员是子类的,且非静态属性是默认初始值的。

package com.base.chapter4;

classBase{

publicstaticint a = 10;

publicint b = 20;

static {

System.out.println("Static Init Base " + a);

//System.out.println("Null Init " + b);

}

public Base() {

System.out.println("Init Base " + this.b);

}

}

//一级子类和基类包含的内容一样

classSuperClassextendsBase{

//静态变量、静态块执行顺序,按书写先后顺序

publicstaticint a1 = getSuperStaticNumber();

publicint b1 = getSuperInstanceNumber();

public SuperClass() {

System.out.println("Init SuperClass" + this.b1);

}

static {

System.out.println("Static Init SuperClass" + a1);

}

public static int getSuperStaticNumber() {

System.out.println("Static member init"); return100;

}

public int getSuperInstanceNumber() {

System.out.println("Instance member init"); return200;

}

}

//二级子类为测试该代码的驱动类

publicclassSubextendsSuperClass{

publicstaticint a2 = getStaticNumber();

publicint b2 = getInstanceNumber();

public Sub() {

System.out.println("Init SubClass " + this.b2);

}

public static int getStaticNumber() {

System.out.println("Static member init Sub");

return1000;

}

public int getInstanceNumber() {

System.out.println("Instance member init Sub");

return2000;

}

static {

System.out.println("Static Init " + a2);

}

// 程序入口,

mainpublic static void main(String args[]) {

new Sub();

}

}

以上内容相关免费视频教程下载:

  • 4
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
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、付费专栏及课程。

余额充值