我的问题是关于静态关键字的一个特殊用法。可以使用静态关键字覆盖不属于任何函数的类中的代码块。例如下面的代码编译:

public class Test {
    private static final int a;    
    static {
        a = 5;
        doSomething(a);
    }
    private static int doSomething(int x) {
        return (x+5);
    }
}

如果删除static关键字,它会报错,因为变量a是final。但是,可以同时删除最终关键字和静态关键字并使其编译。

这两方面都让我感到困惑。我怎么能有一个不属于任何方法的代码段呢?如何调用它呢?一般来说,这种用法的目的是什么?或者更好的是,我可以在哪里找到相关的文档?


当前回答

"final"保证变量必须在对象初始化器代码结束之前初始化。同样,“static final”保证变量将在类初始化代码结束时初始化。从初始化代码中省略“static”将使其变成对象初始化代码;因此,您的变量不再满足其保证。

其他回答

风浪!什么是静态初始化器?

静态初始化器是java类中的一个静态{}代码块,在构造函数或主方法被调用之前只运行一次。

好的!告诉我更多……

是一个代码块静态{…}在任何Java类中。并在类被调用时由虚拟机执行。 不支持返回语句。 不支持参数。 不支持this或super。

嗯,我可以在哪里使用它?

可以用在任何你觉得不错的地方:)就这么简单。但我看到大多数时候它是在做数据库连接、API初始化、日志记录等时使用的。

不要只是吠叫!榜样在哪里?

package com.example.learnjava;

import java.util.ArrayList;

public class Fruit {

    static {
        System.out.println("Inside Static Initializer.");

        // fruits array
        ArrayList<String> fruits = new ArrayList<>();
        fruits.add("Apple");
        fruits.add("Orange");
        fruits.add("Pear");

        // print fruits
        for (String fruit : fruits) {
            System.out.println(fruit);
        }
        System.out.println("End Static Initializer.\n");
    }

    public static void main(String[] args) {
        System.out.println("Inside Main Method.");
    }
}

输出? ?

在静态初始化器内部。 苹果 橙色 梨 结束静态初始化器。 在Main Method内部。

带有静态修饰符的代码块表示类初始化式;如果没有静态修饰符,代码块就是一个实例初始化器。

类初始化器在装入类时(实际上是在解析类时,但这是一个技术术语)按照它们定义的顺序执行(自顶向下,就像简单的变量初始化器一样)。

实例初始化式按照类实例化时定义的顺序执行,即在构造函数代码执行之前,在超级构造函数调用之后。

如果你从int a中删除static,它就变成了一个实例变量,你不能从静态初始化块中访问它。这将导致编译失败,并报错“非静态变量a不能从静态上下文引用”。

如果你也从初始化程序块中删除了static,那么它就变成了实例初始化程序,因此int a在构造时就被初始化了。

"final"保证变量必须在对象初始化器代码结束之前初始化。同样,“static final”保证变量将在类初始化代码结束时初始化。从初始化代码中省略“static”将使其变成对象初始化代码;因此,您的变量不再满足其保证。

您不会将代码写入需要在程序中的任何地方调用的静态块中。如果代码的目的是被调用,那么必须将它放在方法中。

你可以写静态初始化块来初始化静态变量当类加载时,但这段代码可能更复杂。

静态初始化块看起来像一个没有名称、没有参数和没有返回类型的方法。既然你从来不叫它,它就不需要名字了。唯一调用它的时候是虚拟机加载类的时候。

当JVM将类装入内存时,在主方法之前调用静态初始化程序块(按照定义它们的顺序)。它用于有条件地初始化静态变量。

类似地,我们有实例初始化程序块(又名iib),它们在对象实例化时被调用,通常用于去重复构造函数逻辑。

初始化式和构造函数的执行顺序如下:

静态初始化块; 对象初始化器块; 构造函数;