一个java类的加载顺序如下:
- 如果jvm中不存在这个类,则先加载这个类
- 为所有的静态变量分配内存,并设置默认值,count1=0, count2=0, name=null
- 执行static代码块(包括{}和static变量的初始化),相同层级的按代码顺序执行
- 1~3步一个class只执行一次
- 再执行普通代码块
- 执行构造函数
以下面这段代码为例:
public class StaticTest
{
static
{
count1 = 3;
count2 = 4;
System.out.println("static代码块执行");
}
{
System.out.println("普通代码块执行");
}
private static int count1;
private static int count2 = 2;
private static String name = "wang";
public StaticTest()
{
count1++;
count2++;
System.out.println("构造函数: count1=" +count1 + " count2=" + count2);
}
public static void main(String[] args)
{
new StaticTest();
new StaticTest();
}
}
执行结果如下:
static代码块执行
普通代码块执行
构造函数: count1=4 count2=3
普通代码块执行
构造函数: count1=5 count2=4
那么是不是不管怎么样的代码,static代码块都在普通代码块之前执行呢?我们再来看一个例子
public class StaticTest
{
private static StaticTest tester = new StaticTest();
private static int count1;
private static int count2 = 2;
static
{
System.out.println("static代码块执行1: " + "count1=" + count1 + " count2=" + count2 + " tester=" + tester);
count1 = 3;
count2 = 4;
System.out.println("static代码块执行2: " + "count1=" + count1 + " count2=" + count2 + " tester=" + tester);
}
{
System.out.println("普通代码块执行" + "count1=" +count1 + " count2=" + count2 + " tester=" + tester);
}
public StaticTest()
{
count1++;
count2++;
System.out.println("构造函数执行,count1=" +count1 + " count2=" + count2 + " tester=" + tester);
}
public static StaticTest getTester()
{
System.out.println("静态方法执行,count1=" +count1 + " count2=" + count2 + " tester=" + tester);
return tester;
}
public static void main(String[] args)
{
StaticTest.getTester();
System.out.println("-------------------");
new StaticTest();
}
}
看看执行结果:
普通代码块执行count1=0 count2=0
构造函数执行,count1=1 count2=1
static代码块执行count1=3 count2=4
静态方法执行,count1=3 count2=4
-------------------
普通代码块执行count1=3 count2=4
构造函数执行,count1=4 count2=5
貌似普通代码块和构造函数都比static执行的要早,是哪里错了么?其实并没有,解释如下:
- 当类第一次加载的时候,依然为静态变量分配内存,并设置默认值 count1=0,count2=0,tester=null
- 按代码顺序执行static代码块(包括{}和static变量初始化),会先执行tester = new StaticTest() 的new StaticTest()
- new StaticTest()则会执行普通代码块和构造函数,而这时候,这些值还是默认值, 看前两句输出
- 将new StaticTest()赋给tester
- 再执行count2 = 2
- 再执行static代码块,可以看到接下来的两句输出,其中会修改count1和count2的值
- 再执行静态方法
- 第二次new StaticTest()时,类已经加载过了,已经不会执行static相关的语句了,这时只会新实例的普通代码块和构造函数