zoukankan      html  css  js  c++  java
  • 类的加载顺序,静态、非静态、构造方法


    package
    com.fuzi.demo; class SuperLei { public static int a = 10; public int b = 20; static { System.out.println("超类的静态代码块--获取静态参数a的值:" + a); } public SuperLei() { System.out.println("超类的无参构造--获取参数b的值:" + this.b); } } class FuLei extends SuperLei { // 静态变量、静态块执行顺序,按书写先后顺序 public static int a1 = getFuStaticNumber(); public int b1 = getFuInstanceNumber(); public FuLei() { System.out.println("父类的无参构造--获取参数b1的值:" + this.b1); } static { System.out.println("父类的静态代码块--获取静态参数a1的值:" + a1); } public static int getFuStaticNumber() { System.out.println("父类的静态方法被调用"); return 100; } public int getFuInstanceNumber() { System.out.println("父类的方法被调用"); return 200; } } public class ZiLei extends FuLei { public static int a2 = getZiStaticNumber(); public int b2 = getZiInstanceNumber(); public ZiLei() { System.out.println("子类的无参构造--获取参数b2的值:" + this.b2); } public static int getZiStaticNumber() { System.out.println("子类的静态方法被调用"); return 1000; } public int getZiInstanceNumber() { System.out.println("子类的方法被调用"); return 2000; } static { System.out.println("子类的静态代码块--获取静态参数a2的值:" + a2); } public static void main(String args[]) { new ZiLei(); } }

    首先需要知道:

      当执行到子类的构造函数的时候,首先第一个代码是执行super(),哪怕你没有显示的写出来,他也是会去执行父类的实例化

      实例化子类的时候系统默认调用父类无参构造方法super(),即默认会在子类的构造方法中的第一行加上父类的无参构造方法

    总结类加载的顺序: 

    1.加载静态成员/代码块: 

      先从最高父类依次加载其静态成员/代码块(Object的最先);再依次加载到本类的静态成员。 

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

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

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

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

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

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

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

      调用父类的非静态成员(private 除外),也可以像调用自己的一样。 

    3.调用构造方法:

      先从最高父类依次调用其构造方法(Object的最先)也就是上溯下行;默认调用父类空参的,也可在第一行写明调用父类某个带参的。

      再依次到本类的构造方法;构造方法内,也可在第一行写明调用某个本类其它的构造方法。

      注意:如果加载时遇到 override 的成员,可看作是所需创建的类型赋值给当前类型。

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

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

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

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

  • 相关阅读:
    vijos1198:最佳课题选择
    vijos1071:新年趣事之打牌
    vijos1153:猫狗大战
    bzoj3594: [Scoi2014]方伯伯的玉米田
    bzoj2753: [SCOI2012]滑雪与时间胶囊
    bzoj1923: [Sdoi2010]外星千足虫
    bzoj2783: [JLOI2012]树
    bzoj4590: [Shoi2015]自动刷题机
    bzoj4580: [Usaco2016 Open]248
    bzoj4579: [Usaco2016 Open]Closing the Farm
  • 原文地址:https://www.cnblogs.com/java-spring/p/9391196.html
Copyright © 2011-2022 走看看