Java中static关键字的详解

时间:2021-12-06 12:56:37

static关键字可以修饰方法和成员变量(不可以修饰局部变量,但是可以修饰内部类)。

static关键字方便在没有创建对象的情况下来进行调用(方法/变量)。

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

static可以用来修饰类的成员方法、类的成员变量,另外可以编写static代码块来优化程序性能

1.static 修饰成员变量:

static变量也称作静态变量,静态变量和非静态变量的区别是:静态变量被所有的对象所共享,在内存(方法区)中只有一个副本,它当且仅当在类初次加载时会被初始化。而非静态变量是对象所拥有的,在创建对象(构造方法中)的时候被初始化,存在多个副本,各个对象拥有的副本互不影响。想了解java中类的加载过程可以读下一篇博客。里面有详细介绍

2.static修饰方法(例如构造方法)

static方法一般称为静态方法。它不依附于一个具体对象,而可以通过类名进行调用。因此。静态方法中不能访问非静态方法或者非静态成员变量(因为静态方法不依赖对象,而非静态部分需要依赖于对象)。而非静态方法中可以区访问静态的成员变量和静态方法。

3.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
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
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");
    }
}
Java中static关键字的详解
test static
myclass static
person static
person Test
test constructor
person MyClass
myclass constructor
Java中static关键字的详解

  类似地,我们还是来想一下这段代码的具体执行过程。首先加载Test类,因此会执行Test类中的static块。接着执行new MyClass(),而MyClass类还没有被加载,因此需要加载MyClass类。在加载MyClass类的时候,发现MyClass类继承自Test类,但是由于Test类已经被加载了,所以只需要加载MyClass类,那么就会执行MyClass类的中的static块。在加载完之后,就通过构造器来生成对象。而在生成对象的时候,必须先初始化该类的成员变量,因此会执行Test中的Person person = new Person(),而Person类还没有被加载过,因此会先加载Person类并执行Person类中的static块,接着执行父类的构造器,完成了父类的初始化,然后就来初始化自身了,因此会接着执行MyClass中的Person person = new Person(),最后执行MyClass的构造器。