一.简介
1.普通代码块:
类中方法的方法体
2.构造代码块:
构造块会在创建对象时被调用,每次创建时都会被调用,优先于类构造函数执行。
3.静态代码块:
用static{}包裹起来的代码片段,只会执行一次。静态代码块优先于构造块执行。
4.同步代码块:
使用synchronized(){}包裹起来的代码块,在多线程环境下,对共享数据的读写操作是需要互斥进行的,否则会导致数据的不一致性。同步代码块需要写在方法中。
二.静态代码块和构造代码块的异同点
相同点:都是JVM加载类后且在构造函数执行之前执行,在类中可定义多个,一般在代码块中对一些static变量进行赋值。
不同点:静态代码块在非静态代码块之前执行。静态代码块只在第一次new时执行一次,之后不在执行。而非静态代码块每new一次就执行一次。
三.演示
1.普通代码块
普通代码块的存在可以将类的生命周期内只调用一次的代码放在一起,程序调用完之后就立即销毁,这样可以节省栈内存,防止栈溢出。
普通代码块执行的顺序是按照排列的先后顺序。
public class Test1 { 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
2.构造代码块
直接在类中定义且没有加static关键字的代码块称为{}构造代码块。
public class Test1 { { System.out.println("构造代码块先执行~~~"); } public Test1() { System.out.println("构造函数再执行~~~~"); } public static void main(String[] args) { Test1 t = new Test1(); System.out.println("普通代码块最后执行~~~"); } }
运行结果:
构造代码块先执行~~~ 构造函数再执行~~~~ 普通代码块最后执行~~~
3.静态代码块
在java中使用static关键字声明的代码块。静态块用于初始化类,为类的属性初始化。每个静态代码块只会执行一次。
由于JVM在加载类时会执行静态代码块,所以静态代码块先于主方法执行。
public class Test1 { static String s1 = "cjj"; //静态变量 String s2 = "吹静静"; //普通变量 static{//静态变量 System.out.println("静态代码块最先执行~~~" + s1); } {//非静态代码块 System.out.println("构造代码块再执行~~~" + s2); } public Test1() {//构造函数 System.out.println("构造函数再执行~~~~"); } public static void main(String[] args) { Test1 t = new Test1(); System.out.println("普通代码块最后执行~~~"); } }
运行结果:
静态代码块最先执行~~~cjj 构造代码块再执行~~~吹静静 构造函数再执行~~~~ 普通代码块最后执行~~~
注意:
1.静态代码块不能存在于任何方法体内。
2.静态代码块不能直接访问实例变量和实例方法,需要通过类的实例对象来访问
3.在类加载的时候执行一次
4.执行顺序:父类静态 -> 子类静态 -> 父类非静态 -> 父类的构造方法 -> 子类非静态 -> 子类的构造方法
原理:
类在加载阶段,由于 i 处于一个标记值状态,所以实际上是无值的,所以此时不允许直接操作
static int i = 5;
{
i = 7;
}
先将静态变量 i 放入方法区,并且标记一个值为0;在初始化阶段,再检查 i 是否有初值,如果没有初值,则将标记值0赋值进去;如果有初始值,则将初始值设置进去。然后顺次执行静态代码块,将静态变量 i 的值改为7。
static{
i = 7;
}
static int i = 5;
先将静态变量 i 放入方法区,并且标记一个值为0;在初始化阶段,先执行静态代码块,对于 i = 7;并不是将7直接赋值给 i 而是将标记值改为7;初始化静态变量 i ;检查 i 是否有初始值,如果没有初始值,则将标记值7赋值进去;如果有初始值则抛弃标记值,将5赋值进去。