类的生命周期

  • 加载: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");
    }
}

输出结果:

我们来逐个分析,

一开始调用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");
    }
}

执行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
    }
}

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、静态优先于非静态

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

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

到此这篇关于一文搞懂java中类及static关键字执行顺序的文章就介绍到这了,更多相关java static关键字执行顺序内容请搜索Devmax以前的文章或继续浏览下面的相关文章希望大家以后多多支持Devmax!

一文搞懂java中类及static关键字执行顺序的更多相关文章

  1. Swift:实例方法和类型方法

    大家对“实例方法和类型方法”的概念应该不陌生了,在objective-c中很常见。例如:1.实例方法调用的时候,必须先进行实例化一个对象,然后调用init方法。而在Swift中写法有所改变,它使用class和static关键字来标示。

  2. Swift开发必备技巧:static和class的使用

    Swift中表示“类型范围作用域”这一概念有两个不同的关键字,它们分别是static和class。但是在Swift中,这两个关键字却是不能用混的。在非class的类型上下文中,我们统一使用static来描述类型作用域。这包括在enum和struct中表述类型方法和类型属性时。static适用的场景有这些:enum的情况与这个十分类似,就不再列举了。在Swift中class、struct和enum都是可以实现protocol的。

  3. OC和Swift中的static

    在非class的类型上下文中,我们统一使用static来描述类型作用域,class关键字是专门用在class类型的上下文中的,可以用来修饰类方法以及类的计算属性。Swift1.2之后,类也可以通过static关键字拥有类型存储属性了,static相当于classfinal标识符的别名,类中的static属性拥有全局作用域和懒加载属性。

  4. swift的静态属性(方法)和java的不同

    swift在任何情况下静态属性都要加上类名做为前缀,在java中通过实例名使用静态属性是一个warning,在swift中这里是一个错误swift中的static属性===swift中的finalclass====java中的finalstaticswift多出了class属性这个概念,用来表示`可以被子类重写的static属性`,然并卵,我觉得不如直接使用static和finalstatic,少一种概念swift中子类不能覆盖父类的storedproperty,但是可以覆盖父类的computedprop

  5. JokeClient-Swift 仿写学习

    STATIC和CLASSSwift中表示“类型范围作用域”这一概念有两个不同的关键字,它们分别是static和class。在Swift中class,struct和enum都是可以实现某个protocol的。在Swift中编译器不仅不会对AnyObject实例的方法调用做出检查,甚至对于AnyObject的所有方法调用都会返回Optional的结果。摘自:ANY和ANYOBJECTSELECTOR在Swift中没有@selector了,取而代之,从Swift2.2开始我们使用#selector来从暴露给Ob

  6. Swift 学习笔记 [2] 类 结构体 枚举

    错误处理四种错误处理方法传播给调用该函数的代码funcprocesFn()throwtryprocessFn()docatch捕获do{tryexpression}catchpattern1{}catchpattern2{}将错误转成可选值funcprocessFn()throw->Int{}lettry?processFn()defer定义:defer定义的函数内,在此函数完成后,立即再次调用defer定义的代码块。无关的话学了还是要用,第一版Swift文档出来的时候,还凑热闹学了一次,现在都忘得差不多

  7. 在Swift中使用dispatch_once单例模型

    到目前为止,我已经能够得到一个非线程安全模型:在Static结构中包含单例实例应该允许单个实例与单例实例没有冲突,没有复杂的命名模式,它应该使事情相当私密。根据我对Swift的经验,有三种方法来实现支持延迟初始化和线程安全的Singleton模式。dispatch_once传统的Objective-C方法移植到Swift。

  8. Swift中的static func和class func有什么区别?

    我可以在Swift库中看到这些定义:定义为staticfunc的成员函数和定义为func类的另一个函数之间有什么区别?是简单的静态是结构和枚举的静态函数,类和协议的类?有什么其他差别,应该知道吗?在语法本身中有这种区别的理由是什么?一些其他的区别是类函数是动态分派的,可以被子类覆盖。为协议选择了类,因此不必有第三个关键字来表示静态或类。

  9. “dispatch_once_t”在Swift中不可用:使用懒惰初始化的全局变量

    }()_=myGlobal//usingmyGlobalwillinvoketheinitializationcodeonlythefirsttimeitisused.所以我想迁移这个代码.所以在迁移之前:迁移后,按照Apple的指南,代码如下所示:但是当我运行这个访问返回时,我得到以下错误Static.instance!即使Swift2中有效,该代码过于冗长.在Swift3中,Apple强制您通过关闭来使用延迟初始化:

  10. Java利用POI实现导入导出Excel表格

    这篇文章主要为大家详细介绍了Java利用POI实现导入导出Excel表格,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

随机推荐

  1. 基于EJB技术的商务预订系统的开发

    用EJB结构开发的应用程序是可伸缩的、事务型的、多用户安全的。总的来说,EJB是一个组件事务监控的标准服务器端的组件模型。基于EJB技术的系统结构模型EJB结构是一个服务端组件结构,是一个层次性结构,其结构模型如图1所示。图2:商务预订系统的构架EntityBean是为了现实世界的对象建造的模型,这些对象通常是数据库的一些持久记录。

  2. Java利用POI实现导入导出Excel表格

    这篇文章主要为大家详细介绍了Java利用POI实现导入导出Excel表格,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  3. Mybatis分页插件PageHelper手写实现示例

    这篇文章主要为大家介绍了Mybatis分页插件PageHelper手写实现示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  4. (jsp/html)网页上嵌入播放器(常用播放器代码整理)

    网页上嵌入播放器,只要在HTML上添加以上代码就OK了,下面整理了一些常用的播放器代码,总有一款适合你,感兴趣的朋友可以参考下哈,希望对你有所帮助

  5. Java 阻塞队列BlockingQueue详解

    本文详细介绍了BlockingQueue家庭中的所有成员,包括他们各自的功能以及常见使用场景,通过实例代码介绍了Java 阻塞队列BlockingQueue的相关知识,需要的朋友可以参考下

  6. Java异常Exception详细讲解

    异常就是不正常,比如当我们身体出现了异常我们会根据身体情况选择喝开水、吃药、看病、等 异常处理方法。 java异常处理机制是我们java语言使用异常处理机制为程序提供了错误处理的能力,程序出现的错误,程序可以安全的退出,以保证程序正常的运行等

  7. Java Bean 作用域及它的几种类型介绍

    这篇文章主要介绍了Java Bean作用域及它的几种类型介绍,Spring框架作为一个管理Bean的IoC容器,那么Bean自然是Spring中的重要资源了,那Bean的作用域又是什么,接下来我们一起进入文章详细学习吧

  8. 面试突击之跨域问题的解决方案详解

    跨域问题本质是浏览器的一种保护机制,它的初衷是为了保证用户的安全,防止恶意网站窃取数据。那怎么解决这个问题呢?接下来我们一起来看

  9. Mybatis-Plus接口BaseMapper与Services使用详解

    这篇文章主要为大家介绍了Mybatis-Plus接口BaseMapper与Services使用详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  10. mybatis-plus雪花算法增强idworker的实现

    今天聊聊在mybatis-plus中引入分布式ID生成框架idworker,进一步增强实现生成分布式唯一ID,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

返回
顶部