关于Java中代码块的执行顺序

 更新时间:2023年08月22日 09:58:29   作者:ycfxhsw  
这篇文章主要介绍了关于Java中代码块的执行顺序,构造代码块是给所有对象进行统一初始化,而构造函数是给对应的对象初始化,因为构造函数是可以多个的,运行哪个构造函数就会建立什么样的对象,但无论建立哪个对象,都会先执行相同的构造代码块,需要的朋友可以参考下

前言

先说总结:

父类静态字段 —> 父类静态代码块 —> 子类静态字段 —> 子类静态代码块 —> 父类成员变量和非静态块(顺序加载) —> 父类构造函数 —> 子类成员变量和非静态块(顺序加载) —> 子类构造函数

在这里插入图片描述

  • 静态代码块:用staitc声明,jvm加载类时执行,仅执行一次。
  • 构造代码块:类中直接用{}定义,每一次创建对象时执行。
  • 执行顺序优先级:静态块,main(),构造块,构造方法。

构造函数

public HelloA(){//构造函数
}

关于构造函数,以下几点要注意:

  • 对象一建立,就会调用与之相应的构造函数,也就是说,不建立对象,构造函数时不会运行的。
  • 构造函数的作用是用于给对象进行初始化。
  • 一个对象建立,构造函数只运行一次,而一般方法可以被该对象调用多次。

构造代码块

{//构造代码块    
}

关于构造代码块,以下几点要注意:

  • 构造代码块的作用是给对象进行初始化。
  • 对象一建立就运行构造代码块了,而且优先于构造函数执行。这里要强调一下,有对象建立,才会运行构造代码块,类不能调用构造代码块的,而且构造代码块与构造函数的执行顺序是前者先于后者执行。
  • 构造代码块与构造函数的区别是:构造代码块是给所有对象进行统一初始化,而构造函数是给对应的对象初始化,因为构造函数是可以多个的,运行哪个构造函数就会建立什么样的对象,但无论建立哪个对象,都会先执行相同的构造代码块。也就是说,构造代码块中定义的是不同对象共性的初始化内容。

静态代码块

static {//静态代码块    
}

关于静态代码块,要注意的是:

  • 它是随着类的加载而执行,只执行一次,并优先于主函数。具体说,静态代码块是由类调用的。类调用时,先执行静态代码块,然后才执行主函数的。
  • 静态代码块其实就是给类初始化的,而构造代码块是给对象初始化的。
  • 静态代码块中的变量是局部变量,与普通函数中的局部变量性质没有区别。
  • 一个类中可以有多个静态代码块。
public class Test {
    staitc int cnt=6;
    static {
        cnt += 9;
    }
    public static void main(String[] args) {
        System.out.println(cnt);
    }
    static {
        cnt /= 3;
    }
}
// 5

Java类初始化顺序

对于一个类的情况

public class HelloA {
    public HelloA(){//构造函数
        System.out.println("A的构造函数");    
    }
    {//构造代码块
        System.out.println("A的构造代码块");    
    }
    static {//静态代码块
        System.out.println("A的静态代码块");        
    }
    public static void main(String[] args) {
    }
}
// A的静态代码块
public class HelloA {
    public HelloA(){//构造函数
        System.out.println("A的构造函数");    
    }
    {//构造代码块
        System.out.println("A的构造代码块");    
    }
    static {//静态代码块
        System.out.println("A的静态代码块");        
    }
    public static void main(String[] args) {
        HelloA a=new HelloA();    
    }
}
// A的静态代码块
// A的构造代码块
// A的构造函数
public class HelloA {
    public HelloA(){//构造函数
        System.out.println("A的构造函数");    
    }
    {//构造代码块
        System.out.println("A的构造代码块");    
    }
    static {//静态代码块
        System.out.println("A的静态代码块");        
    }
    public static void main(String[] args) {
        HelloA a=new HelloA();
        HelloA b=new HelloA();
    }
}
// A的静态代码块
// A的构造代码块
// A的构造函数
// A的构造代码块
// A的构造函数

对于一个类而言,按照如下顺序执行:

  • 执行静态代码块
  • 执行构造代码块
  • 执行构造函数

对于静态变量、静态初始化块、变量、初始化块、构造器,它们的初始化顺序依次是静态变量、静态初始化块)>(变量、初始化块)>构造器。

public class InitialOrderTest {
        /* 静态变量 */
    public static String staticField = "静态变量";
        /* 变量 */
    public String field = "变量";
        /* 静态初始化块 */
    static {
        System.out.println( staticField );
        System.out.println( "静态初始化块" );
    }
        /* 初始化块 */
    {
        System.out.println( field );
        System.out.println( "初始化块" );
    }
        /* 构造器 */
    public InitialOrderTest(){
        System.out.println( "构造器" );
    }
    public static void main( String[] args ){
        new InitialOrderTest();
    }
}
// 静态变量
// 静态初始化块
// 变量
// 初始化块
// 构造器

对于继承情况

public class HelloA {
    public HelloA(){//构造函数
        System.out.println("A的构造函数");    
    }
    {//构造代码块
        System.out.println("A的构造代码块");    
    }
    static {//静态代码块
        System.out.println("A的静态代码块");        
    }
}
public class HelloB extends HelloA{
    public HelloB(){//构造函数
        System.out.println("B的构造函数");    
    }
    {//构造代码块
        System.out.println("B的构造代码块");    
    }
    static {//静态代码块
        System.out.println("B的静态代码块");        
    }
    public static void main(String[] args) {
        HelloB b=new HelloB();        
    }
}
// A的静态代码块
// B的静态代码块
// A的构造代码块
// A的构造函数
// B的构造代码块
// B的构造函数

当涉及到继承时,按照如下顺序执行:

  • 执行父类的静态代码块,并初始化父类静态成员变量
  • 执行子类的静态代码块,并初始化子类静态成员变量
  • 执行父类的构造代码块,执行父类的构造函数,并初始化父类普通成员变量
  • 执行子类的构造代码块, 执行子类的构造函数,并初始化子类普通成员变量
class Parent {
        /* 静态变量 */
    public static String p_StaticField = "父类--静态变量";
         /* 变量 */
    public String p_Field = "父类--变量";
    protected int i = 9;
    protected int j = 0;
        /* 静态初始化块 */
    static {
        System.out.println( p_StaticField );
        System.out.println( "父类--静态初始化块" );
    }
        /* 初始化块 */
    {
        System.out.println( p_Field );
        System.out.println( "父类--初始化块" );
    }
        /* 构造器 */
    public Parent(){
        System.out.println( "父类--构造器" );
        System.out.println( "i=" + i + ", j=" + j );
        j = 20;
    }
}
public class SubClass extends Parent {
         /* 静态变量 */
    public static String s_StaticField = "子类--静态变量";
         /* 变量 */
    public String s_Field = "子类--变量";
        /* 静态初始化块 */
    static {
        System.out.println( s_StaticField );
        System.out.println( "子类--静态初始化块" );
    }
       /* 初始化块 */
    {
        System.out.println( s_Field );
        System.out.println( "子类--初始化块" );
    }
       /* 构造器 */
    public SubClass(){
        System.out.println( "子类--构造器" );
        System.out.println( "i=" + i + ",j=" + j );
    }
        /* 程序入口 */
    public static void main( String[] args ){
        System.out.println( "子类main方法" );
        new SubClass();
    }
}
// 父类--静态变量
// 父类--静态初始化块
// 子类--静态变量
// 子类--静态初始化块
// 子类main方法
// 父类--变量
// 父类--初始化块
// 父类--构造器
// i=9, j=0
// 子类--变量
// 子类--初始化块
// 子类--构造器
// i=9,j=20

子类的静态变量和静态初始化块的初始化是在父类的变量、初始化块和构造器初始化之前就完成了。

静态变量、静态初始化块,变量、初始化块初始化了顺序取决于它们在类中出现的先后顺序。

到此这篇关于关于Java中代码块的执行顺序的文章就介绍到这了,更多相关Java代码块执行内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java删除String中空格的多种解决方法汇总

    Java删除String中空格的多种解决方法汇总

    在Java中从字符串中删除空格有很多不同的方法,如trim,replaceAll等,下面这篇文章主要给大家介绍了关于Java删除String中空格的多种解决方法,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-06-06
  • 解决SpringMVC @RequestMapping不设置value出现的问题

    解决SpringMVC @RequestMapping不设置value出现的问题

    这篇文章主要介绍了解决SpringMVC @RequestMapping不设置value出现的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-08-08
  • 零基础搭建boot+MybatisPlus的详细教程

    零基础搭建boot+MybatisPlus的详细教程

    这篇文章主要介绍了零基础搭建boot+MybatisPlus,首先需要创建数据库表和创建boot项目使用mybatisplus操作数据库,本文通过示例代码给大家介绍的非常详细,需要的朋友可以参考下
    2022-03-03
  • JVM 心得 OOM时的堆信息获取方法与分析

    JVM 心得 OOM时的堆信息获取方法与分析

    下面小编就为大家带来一篇JVM 心得 OOM时的堆信息获取方法与分析。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-10-10
  • java实现动态代理示例分享

    java实现动态代理示例分享

    动态代理作为代理模式的一种扩展形式,广泛应用于框架(尤其是基于AOP的框架)的设计与开发,本文将通过实例来讲解Java动态代理的实现过程。
    2014-03-03
  • 为了多次读取ServletInputStream引发的一系列问题

    为了多次读取ServletInputStream引发的一系列问题

    这篇文章主要介绍了为了多次读取ServletInputStream引发的一系列问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-10-10
  • 详解Spring Boot Admin监控服务上下线邮件通知

    详解Spring Boot Admin监控服务上下线邮件通知

    本篇文章主要介绍了详解Spring Boot Admin监控服务上下线邮件通知,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-12-12
  • 浅谈Java中的参数传递问题

    浅谈Java中的参数传递问题

    这篇文章主要介绍了Java中的参数传递问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-04-04
  • SpringBoot入坑笔记之spring-boot-starter-web 配置文件的使用

    SpringBoot入坑笔记之spring-boot-starter-web 配置文件的使用

    本篇向小伙伴介绍springboot配置文件的配置,已经全局配置参数如何使用的。需要的朋友跟随脚本之家小编一起学习吧
    2018-01-01
  • spring cloud zuul增加header传输的操作

    spring cloud zuul增加header传输的操作

    这篇文章主要介绍了spring cloud zuul增加header传输的操作,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-06-06

最新评论