java 代码块执行顺序

部分内容也是引入别人的,加了自己的内容和验证,修改了部分描述不准确的地方。

 

java中用{}括起来的称为代码块,代码块可分为以下四种:

.简介

1.普通代码块:

类中方法的方法体

2.构造代码块

构造块会在创建对象时被调用,每次创建时都会被调用,优先于类构造函数执行。

3.静态代码块:

static{}包裹起来的代码片段,只会执行一次。静态代码块优先于构造块执行。

4.同步代码块:

使用synchronized(){}包裹起来的代码块,在多线程环境下,对共享数据的读写操作是需要互斥进行的,否则会导致数据的不一致性。同步代码块需要写在方法中。

.静态代码块和构造代码块的异同点

相同点:都是JVM加载类后且在构造函数执行之前执行,在类中可定义多个,一般在代码块中对一些static变量进行赋值。

不同点:静态代码块在非静态代码块之前执行。静态代码块只执行一次,之后不在执行。而非静态代码块每new一次就执行一次。

 

普通代码块:在方法或语句中出现的{}就称为普通代码块。普通代码块和一般语句的执行顺序由他们在代码中出现的次序决定,先出现先执行。

public class Test {

  public static void main(String[] args) {

    {

      int x = 3;

      System.out.println("普通代码块内的变量x=" + x);

    }

    int x = 1;

    System.out.println("主方法内的变量x=" + x);

    {

      int y = 7;

      System.out.println("普通代码块内的变量y=" + y);

    }

  }

}

/*

 * 运行结果  普通代码块内的变量x=3

 *      主方法内的变量x=1

 *      普通代码块内的变量y=7

 */

 

构造代码块:直接在类中定义且没有加static关键字的代码块称为{}构造代码块。构造代码块在创建对象时被调用,每次创建对象都会被调用,并且构造代码块的执行次序优先于类构造函数。如果存在多个构造代码块,执行顺序由他们在代码中出现的次序决定,先出现先执行。

public class Test1 {

    {

        System.out.println("第一构造块");

    }

    public Test1(int i) {

        System.out.println("" + i + "次调用" + "构造方法");

    }

    {

        System.out.println("第二构造块");

    }

    public static void main(String[] args) {

        new Test1(0);

        new Test1(1);

        new Test1(2);

    }

}

/*

 * 执行结果 第一构造块

 *      第二构造块

 *      0次调用构造方法

 *      第一构造块

 *      第二构造块

 *      1次调用构造方法

 *      第一构造块

 *      第二构造块

 *      2次调用构造方法

 */

 

静态代码块:java中使用static关键字声明的代码块。静态块用于初始化类,为类的属性初始化。每个静态代码块只会执行一次。由于JVM在加载类时会执行静态代码块,所以静态代码块先于主方法执行。

如果类中包含多个静态代码块,那么将按照"先定义的代码先执行,后定义的代码后执行"

注意:

1.静态代码块不能存在于任何方法体内

2.静态代码块不能直接访问实例变量和实例方法,需要通过类的实例对象来访问。

 

public class Test3 {

    public static String STATIC_FIELD = "静态属性";

    // 静态块

    static {

        System.out.println(STATIC_FIELD);

        System.out.println("静态代码块1");

    }

    public String field = "非静态属性";

    // 非静态块

    {

        System.out.println(field);

        System.out.println("非静态代码块2");

    }

    public InitOderTest() {

        System.out.println("无参构造函数");

    }

    public static void main(String[] args) {

        InitOderTest test = new InitOderTest();

    }

    // 非静态块

    {

        System.out.println(field);

        System.out.println("非静态代码块1");

    }

    // 静态块

    static {

        System.out.println(STATIC_FIELD);

        System.out.println("静态代码块2");

    }

}

/*

 * 运行结果 静态属性

 *      静态代码块1

 *      静态属性

 *      静态代码块2

 *      非静态属性

 *      非静态代码块2

 *      非静态属性

 *      非静态代码块1

 *      无参构造函数

 */

我们可以看出:静态块不论new或不new对象,总是会执行,且最先执行的,并且只执行一次;(然后new对象时)第二执行的是构造块;第三执行的是构造方法。

 

如果一个类有父类,则先执行父类的静态代码块,然后执行自己的静态代码块,然后new对象时,先执行父类的构造块,然后是父类的构造方法,再次是自己的构造块,再次是自己的构造方法。

例:

public class CodeBlockTest extends B {

 

    public static int i=0;

    public static void main(String[] args) {

 

        {

            System.out.println("main方法中第一行代码,还没new 对象呢 ");

        }

        new CodeBlockTest();

        new CodeBlockTest();

    }

 

    {

        System.out.println("普通代码块A");

    }

 

    static {

        System.out.println("静态块A");

   

    }

 

    public CodeBlockTest() {

        super();

        i++;

        System.out.println("构造器A" + i);

    }

 

}

 

class B {

 

    public B() {

        super();

        System.out.println("构造器B");

    }

 

    {

        System.out.println("普通的代码块B");

    }

    static {

        System.out.println("静态代码块B");

    }

 

}

运行结果:

静态代码块B

静态块A

main方法中第一行代码,还没new 对象呢

普通的代码块B

构造器B

普通代码块A

构造器A1

普通的代码块B

构造器B

普通代码块A

构造器A2

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值