当前位置: 首页 > news >正文

一篇文章搞懂java中类以及static关键字执行顺序

一篇文章搞懂java中类以及static关键字执行顺序

这里写目录标题

  • 一篇文章搞懂java中类以及static关键字执行顺序
    • 类的生命周期
    • static关键字
    • static关键字执行顺序
    • 继承中的static执行顺序
        • 例子一
        • 例子二(重点)
    • 总结

我们先看看类的生命周期

类的生命周期

image-20220831171306831

  • 加载:classpath、jar包、网络、某个磁盘位置下的类的class二进制字节流读进来,在内存中生成一个代表这个类的java.lang.Class对象放入元空间(方法区,永久代),此阶段我们程序员可以干预,我们可以自定义类加载器来实现类的加载;
  • 验证:验证Class文件的字节流中包含的信息符合《Java虚拟机规范》的全部约束要求,保证虚拟机的安全;
  • 准备:类变量赋默认初始值,int为0,long为0L,boolean为false,引用类型为null;常量赋正式值;
  • 解析:把符号引用翻译为直接引用;
  • 初始化:当我们new一个类的对象,访问一个类的静态属性,修改一个类的静态属性,调用一个类的静态方法,用反射API对一个类进行调用,初始化当前类,其父类也会被初始化… 那么这些都会触发类的初始化;

静态变量的初始化有两种途径: (1)在静态变量的声明处进行初始化 (2)在静态代码块中进行初始化

  • 使用:使用这个类;
  • 卸载:
    1.该类所有的实例都已经被GC,也就是JVM中不存在该Class的任何实例;
    2.加载该类的ClassLoader已经被GC;
    3.该类的java.lang.Class 对象没有在任何地方被引用,如不能在任何地方通过反射访问该类的方法;

接着我们来了解static关键字的作用和使用条件

static关键字

static关键字修饰的数据存储在我们的方法区中的静态常量池中,static可以修饰方法、变量和代码块

static修饰方法:

  • 指定不需要实例化就可以激活的一个方法。
  • this关键字不能再static方法中使用
  • 静态方法中不能使用非静态方法
  • 非静态方法可以调用静态方法。

static修饰变量:

  • 指定变量被所有对象共享,即所有实例都可以使用该变量。
  • 变量属于这个类。

static修饰代码块:

  • 无论放在哪里,都是放在main方法运行的。
  • 通常用于初始化静态变量,静态代码块属于类
  • 不可以省略,没加static认为是构造代码块

static关键字执行顺序

先记住几个准则

  • 实例化对象前,先加载类(对象载入之前,一定要是类先被载入)
  • 类(或者静态变量和静态代码块)在生命周期结束前,只执行一次
  • 静态变量(属性)和静态代码块谁先声明谁先执行
  • 非静态变量(属性)和非静态代码块谁先声明谁先执行
  • 静态构造代码块是和类同时加载的,构造代码块是在实例化之后执行构造方法之前执行的
  • 构造方法是在构造代码块执行完之后才执行的。
  • 静态方法属于类的,加载完类就可以调用静态方法(可以执行多次);非静态方法是属于对象的,加载完对象就可以调用非静态方法。
  • 每创建一个对象,即每载入一个对象,非静态代码块都执行一次。执行类对象的载入之前就会调用

我们来通过一个例子来验证以下上面的观点

InitializeDemo.java

package com.qcby.demo.staticDemo;

/**
 * @ClassName TRRest
 * @Description TODO
 * @Author heaboy@heaboy.com
 * @Version 1.0.0
 */
public class InitializeDemo {
    private static int k = 1;
    private static InitializeDemo t1 = new InitializeDemo("t1");
    private static InitializeDemo t2 = new InitializeDemo("t2");
    private static int i = print("i");
    private static int n = 99;

    {
        print("初始化块");
        j=100;
    }
    public InitializeDemo(String str) {
        System.out.println((k++) + ":" + str + "   i=" + i + "    n=" + n);
        ++i;
        ++n;

    }
    static {
        print("静态块");
        n=100;
    }
    private int j = print("j");
    public static int print(String str) {
        System.out.println((k++) + ":" + str + "   i=" + i + "    n=" + n);
        ++n;
        return ++i;
    }

    public static void main(String[] args) {
        InitializeDemo test = new InitializeDemo("test");
    }

}

输出结果

image-20220831111748289

我们来逐个分析,

一开始调用main方法,main方法内实例化InitializeDemo的对象,在对象载入之前,一定要是类先被载入

所以我们先加载InitializeDemo类,加载类的同时,会加载静态变量和静态代码块,但是是按顺序执行,且只执行一次

先加载如下静态变量

private static int k = 1;

加载如下静态变量的时候,发现要去加载类,由于类以及加载了,所以会加载这个对象,这个对象加载前,会执行非静态代码块

private static InitializeDemo t1 = new InitializeDemo("t1");

此时也就是

1:初始化块 i=0 n=0

接着执行

private int j = print("j");

2:j i=1 n=1

接着执行构造方法,

public InitializeDemo(String str) {
        System.out.println((k++) + ":" + str + "   i=" + i + "    n=" + n);
        ++i;
        ++n;

    }

3:t1 i=2 n=2

t1的实例化执行结束,接着执行t2的实例化

private static InitializeDemo t2 = new InitializeDemo("t2");

结果和上述一致,按非静态代码块和非静态属性然后构造方法方法的顺序执行

4:初始化块 i=3 n=3

5:j i=4 n=4

6:t2 i=5 n=5

两个静态属性(实例化)执行完,执行如下代码

private static int i = print("i");

7:i i=6 n=6

接着执行下面的代码,此时n变成了99

private static int n = 99;

接着执行静态代码块

static {
        print("静态块");
        n=100;
    }

8:静态块 i=7 n=99

类加载完毕,执行test对象的载入,参考t1,t2的实例化,按非静态代码块和非静态属性然后构造方法方法的顺序执行

9:初始化块 i=8 n=100

10:j i=9 n=101

11:test i=10 n=102

继承中的static执行顺序

例子一

package com.qcby.demo.oop;

public class Test3 extends Base {
    static {
        System.out.println("test static");
    }
    public Test3(){
        System.out.println("test constructor");
    }

    public static void main(String[] args) {
        new Test3();
    }
}
class Base{
    static {
        System.out.println("Base static");
    }
    public Base(){
        System.out.println("Base constructor");
    }
}

image-20220606215857025

执行Test3的构造方法,要先加载Test3的类加载,由于Test3继承于Base,所以他要先加载父类Base,

所以先Base static 后test static 在执行子类的构造方法的时候,要先执行父类的构造方法,如果是多级继承,会先执行最顶级父类的构造方法,然后依次执行各级个子类的构造方法。 所以先执行Base constructor后执行test constructor结果就如上图

例子二(重点)

package com.qcby.demo.oop;

public class MyTest {
    MyPerson person = new MyPerson("test");//这里可以理解为成员变量辅助,,要先把MyPerson先加载到jvm中
    static {
        System.out.println("test static");//1
    }

    public MyTest() {
        System.out.println("test constructor");//5
    }

    public static void main(String[] args) {//main方法在MyTest类中,使用mian方法先加载MyTest的静态方法,不调用其他,
        MyClass myClass =new MyClass();//对象创建的时候,会加载对应的成员变量
    }
}
class MyPerson {
    static {
        System.out.println("person static");//3
    }

    public MyPerson(String str) {
        System.out.println("person " + str);//4  6
    }
}
class MyClass extends MyTest {
    MyPerson person = new MyPerson("class");//这里可以理解为成员变量辅助,要先把MyPerson先加载到jvm中
    static {
        System.out.println("class static");//2
    }
    
    public MyClass() {
        //默认super()
        System.out.println("class constructor");//7
    }
}

image-20220606215905442

1.先看main方法,main方法回先加载对应的类,此时MyTest类和其静态的变量,方法和代码块会随类的加载而开辟空间。static是属于类的。所以test static优先执行,且此时MyTest类的其他语句不执行。

2.mian方法中调用了MyClass myClass =new MyClass(),实例化了一个MyClass类的对象,这时候会初始化对象的成员变量和调用对象的构造函数,而MyClass类继承于MyTest类,在加载MyClass类前,会先调用MyTest类,但是MyTest类以及其静态的变量,方法和代码块已经加载(在类的生命周期只执行一次),所以返回到子类(MyClass类)的加载,这时候会调用MyClass类的静态的变量,方法和代码块。所以class static第二个执行。

3.MyClass类加载完后,应该接着调用MyClass类的构造方法,在调用子类的构造方法前,会默认调用父类的无参构造方法(super()省略),调用父类的无参构造方法,相当于实例化父类的对象,这时候会先初始化对象的成员变量,这里的MyPerson person = new MyPerson(“test”);就相当于成员变量(属于对象,在对象的实例化的时候才加载),于是会加载MyPerson类和其静态的变量,方法和代码块。所以person static第三个执行

4.加载完MyPerson类和其静态的变量,方法和代码块后,会调用MyPerson类和的有参构造方法,即person test第四个执行

5.MyPerson类和的有参构造方法执行结束,返回父类MyTest,父类调用构造方法,即test constructor第五个执行

6.父类MyTest构造方法执行结束,返回子类,子类再调用构造方法前,先初始化对象的成员变量MyPerson person = new MyPerson(“class”);,这时候会先先加载MyPerson 和其静态的变量,方法和代码块。由于上述类以及加载,所以直接执行其有参构造方法,即person class第六个执行

7.MyPerson类和的有参构造方法执行结束,返回子类MyClass,子类调用构造方法,即class constructor第七个执行

父类static代码块–>子类static代码块–>父类普通代码块(成员对象属性初始化)–>父类构造方法–>子类普通代码块(成员对象属性初始化–>子类构造方法

总结

类加载顺序的三个原则是

1、父类优先于子类
2、属性和代码块(看先后顺序)优先于构造方法
3、静态优先于非静态

类加载顺序为(默认变量卸载代码块前)

父类静态变量->父类静态语句块->子类静态变量->子类静态语句块->父类普通成员变量->父类动态语句块->父类构造器->子类普通成员变量->子类动态语句块->子类构造器

相关文章:

  • 新手设计师一定要逛这几个网站
  • 多目标优化算法|基于拥挤距离的有效多目标人工蜂鸟算法,用于解决工程设计问题(Matlab代码实现)
  • yara分析
  • 早上一上班发现产品出现重大事故,作为产品经理该怎么办?
  • PaddleHub开源模型400+,三行代码也可实现无限AI创意梦想!
  • [Linux] CE知识随笔含Ansible、防火墙、VIM、其他服务
  • java架构知识点-中间件
  • 基于SSM的视频管理系统【完整项目源码】
  • 做到年收入一百万需要怎样做?
  • 人工神经网络连接方式,全连接神经网络作用
  • RabbitMq消息队列
  • 神经网络计算机的用途是,神经网络计算机的应用
  • visual studio快捷键
  • gif制作动图教你一键搞定,图片转gif和视频转gif怎么制作
  • 国产数据库百家争鸣,百花齐放有感
  • 《深入 React 技术栈》
  • 【跃迁之路】【519天】程序员高效学习方法论探索系列(实验阶段276-2018.07.09)...
  • HTTP传输编码增加了传输量,只为解决这一个问题 | 实用 HTTP
  • use Google search engine
  • 案例分享〡三拾众筹持续交付开发流程支撑创新业务
  • 翻译:Hystrix - How To Use
  • 分享几个不错的工具
  • 给github项目添加CI badge
  • 记录一下第一次使用npm
  • 记一次和乔布斯合作最难忘的经历
  • 买一台 iPhone X,还是创建一家未来的独角兽?
  • 一个JAVA程序员成长之路分享
  • 阿里云ACE认证学习知识点梳理
  • 你学不懂C语言,是因为不懂编写C程序的7个步骤 ...
  • #include<初见C语言之指针(5)>
  • (2020)Java后端开发----(面试题和笔试题)
  • (C++17) optional的使用
  • (LeetCode C++)盛最多水的容器
  • (Mac上)使用Python进行matplotlib 画图时,中文显示不出来
  • (切换多语言)vantUI+vue-i18n进行国际化配置及新增没有的语言包
  • (生成器)yield与(迭代器)generator
  • (十六)Flask之蓝图
  • (四)Tiki-taka算法(TTA)求解无人机三维路径规划研究(MATLAB)
  • (完整代码)R语言中利用SVM-RFE机器学习算法筛选关键因子
  • (万字长文)Spring的核心知识尽揽其中
  • (译)2019年前端性能优化清单 — 下篇
  • (转)为C# Windows服务添加安装程序
  • .apk文件,IIS不支持下载解决
  • .Net MVC4 上传大文件,并保存表单
  • .NET MVC之AOP
  • .Net多线程总结
  • .NET中GET与SET的用法
  • /proc/vmstat 详解
  • /var/log/cvslog 太大
  • @Autowired和@Resource的区别
  • @JsonSerialize注解的使用
  • []使用 Tortoise SVN 创建 Externals 外部引用目录
  • [Big Data - Kafka] kafka学习笔记:知识点整理
  • [C++] Windows中字符串函数的种类
  • [EFI]Lenovo ThinkPad X280电脑 Hackintosh 黑苹果引导文件