Java static 关键字

加载顺序:

  • 父类的 static
  • 子类的 static
  • 父类的 成员变量和 构造函数
  • 子类的 成员变量和 构造函数

被 static 关键字修饰的方法或者变量不需要依赖于对象来进行访问,只要类被加载了,就可以通过类名去进行访问。static 是不允许用来修饰局部变量。

问题来了,带有 static 关键字的类,执行的先后顺序是什么呢?先看几个例子。

一、下面这段代码的输出结果是什么?

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
public class Test extends Base{
static{
System.out.println("test static");
}
public Test(){
System.out.println("test constructor");
}
public static void main(String[] args) {
new Test();
}
}
class Base{
static{
System.out.println("base static");
}
public Base(){
System.out.println("base constructor");
}
}
1
2
3
4
base static
test static
base constructor
test constructor

至于为什么是这个结果,我们先不讨论,先来想一下这段代码具体的执行过程,

  1. 在执行开始,先要寻找到 main 方法,因为 main 方法是程序的入口,但是在执行 main 方法之前,必须先加载 Test 类
  2. 在加载 Test 类的时候发现 Test 类继承自 Base 类,因此会转去先加载 Base 类,在加载 Base 类的时候,发现有 static 块,便执行了 static 块。
  3. 在 Base 类加载完成之后,便继续加载 Test 类,然后发现 Test 类中也有 static 块,便执行 static 块。在加载完所需的类之后,便开始执行 main 方法。
  4. 在 main 方法中执行 new Test() 的时候会先调用父类的构造器,然后再调用自身的构造器。因此,便出现了上面的输出结果。

二、下面这段代码的输出结果是什么?

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
public class Test {
Person person = new Person("Test");
static{
System.out.println("test static");
}
public Test() {
System.out.println("test constructor");
}
public static void main(String[] args) {
new MyClass();
}
}

class Person{
static{
System.out.println("person static");
}
public Person(String str) {
System.out.println("person "+str);
}
}

class MyClass extends Test {
Person person = new Person("MyClass");
static{
System.out.println("myclass static");
}

public MyClass() {
System.out.println("myclass constructor");
}
}
1
2
3
4
5
6
7
test static
myclass static
person static
person Test
test constructor
person MyClass
myclass constructor

类似地,我们还是来想一下这段代码的具体执行过程。

  1. 首先加载Test类,因此会执行Test类中的static块。接着执行new MyClass(),而MyClass类还没有被加载,因此需要加载MyClass类。
  2. 在加载MyClass类的时候,发现MyClass类继承自Test类,但是由于Test类已经被加载了,所以只需要加载MyClass类,那么就会执行MyClass类的中的static块。
  3. 在加载完之后,就通过构造器来生成对象。而在生成对象的时候,必须先初始化父类的成员变量,因此会执行Test中的Person person = new Person()
  4. 而Person类还没有被加载过,因此会先加载Person类并执行Person类中的static块,接着执行父类的构造器,完成了父类的初始化,然后就来初始化自身了
  5. 因此会接着执行MyClass中的Person person = new Person(),最后执行MyClass的构造器。

三、这段代码的输出结果是什么?

1
2
3
4
5
6
7
8
9
10
11
12
13
public class Test {

static{
System.out.println("test static 1");
}
public static void main(String[] args) {

}

static{
System.out.println("test static 2");
}
}

虽然在main方法中没有任何语句,但是还是会输出,原因上面已经讲述过了。

另外,static块可以出现类中的任何地方(只要不是方法内部,记住,任何方法内部都不行),并且执行是按照static块的顺序执行的。

参考摘抄自https://www.cnblogs.com/dolphin0520/p/3799052.html


执行顺序:

  1. main函数所在的类,若有父类,先加载父类的静态代码块和静态赋值、之后再加载子类的【只执行一次】
  2. 回到main函数,执行相应的对象初始化。调用类的构造函数前,先初始化父类中的成员变量,再是父类的构造函数,再是子类的成员变量-子类的构造函数

Java static 关键字
https://flepeng.github.io/021-Java-Java-static-关键字/
作者
Lepeng
发布于
2021年5月14日
许可协议