ThinkChat🤖让你学习和工作更高效,注册即送10W Token,即刻开启你的AI之旅 广告
### [静态数据的初始化](https://lingcoder.gitee.io/onjava8/#/book/06-Housekeeping?id=%e9%9d%99%e6%80%81%e6%95%b0%e6%8d%ae%e7%9a%84%e5%88%9d%e5%a7%8b%e5%8c%96) 无论创建多少个对象,静态数据都只占用一份存储区域。**static**关键字不能应用于局部变量,所以只能作用于属性(字段、域)。如果一个字段是静态的基本类型,你没有初始化它,那么它就会获得基本类型的标准初值。如果它是对象引用,那么它的默认初值就是**null**。 如果在定义时进行初始化,那么静态变量看起来就跟非静态变量一样。 下面例子显示了静态存储区是何时初始化的: ~~~ // housekeeping/StaticInitialization.java // Specifying initial values in a class definition class Bowl { Bowl(int marker) { System.out.println("Bowl(" + marker + ")"); } void f1(int marker) { System.out.println("f1(" + marker + ")"); } } class Table { static Bowl bowl1 = new Bowl(1); Table() { System.out.println("Table()"); bowl2.f1(1); } void f2(int marker) { System.out.println("f2(" + marker + ")"); } static Bowl bowl2 = new Bowl(2); } class Cupboard { Bowl bowl3 = new Bowl(3); static Bowl bowl4 = new Bowl(4); Cupboard() { System.out.println("Cupboard()"); bowl4.f1(2); } void f3(int marker) { System.out.println("f3(" + marker + ")"); } static Bowl bowl5 = new Bowl(5); } public class StaticInitialization { public static void main(String[] args) { System.out.println("main creating new Cupboard()"); new Cupboard(); System.out.println("main creating new Cupboard()"); new Cupboard(); table.f2(1); cupboard.f3(1); } static Table table = new Table(); static Cupboard cupboard = new Cupboard(); } ~~~ 输出: ~~~ Bowl(1) Bowl(2) Table() f1(1) Bowl(4) Bowl(5) Bowl(3) Cupboard() f1(2) main creating new Cupboard() Bowl(3) Cupboard() f1(2) main creating new Cupboard() Bowl(3) Cupboard() f1(2) f2(1) f3(1) ~~~ **Bowl**类展示类的创建,而**Table**和**Cupboard**在它们的类定义中包含**Bowl**类型的静态数据成员。注意,在静态数据成员定义之前,**Cupboard**类中先定义了一个**Bowl**类型的非静态成员**b3**。 由输出可见,静态初始化只有在必要时刻才会进行。如果不创建**Table**对象,也不引用**Table.bowl1**或**Table.bowl2**,那么静态的**Bowl**类对象**bowl1**和**bowl2**永远不会被创建。只有在第一个 Table 对象被创建(或被访问)时,它们才会被初始化。此后,静态对象不会再次被初始化。 初始化的顺序先是静态对象(如果它们之前没有被初始化的话),然后是非静态对象,从输出中可以看出。要执行`main()`方法,必须加载**StaticInitialization**类,它的静态属性**table**和**cupboard**随后被初始化,这会导致它们对应的类也被加载,而由于它们都包含静态的**Bowl**对象,所以**Bowl**类也会被加载。因此,在这个特殊的程序中,所有的类都会在`main()`方法之前被加载。实际情况通常并非如此,因为在典型的程序中,不会像本例中所示的那样,将所有事物通过**static**联系起来。 概括一下创建对象的过程,假设有个名为**Dog**的类: 1. 即使没有显式地使用**static**关键字,构造器实际上也是静态方法。所以,当首次创建**Dog**类型的对象或是首次访问**Dog**类的静态方法或属性时,Java 解释器必须在类路径中查找,以定位**Dog.class**。 2. 当加载完**Dog.class**后(后面会学到,这将创建一个**Class**对象),有关静态初始化的所有动作都会执行。因此,静态初始化只会在首次加载**Class**对象时初始化一次。 3. 当用`new Dog()`创建对象时,首先会在堆上为**Dog**对象分配足够的存储空间。 4. 分配的存储空间首先会被清零,即会将**Dog**对象中的所有基本类型数据设置为默认值(数字会被置为 0,布尔型和字符型也相同),引用被置为**null**。 5. 执行所有出现在字段定义处的初始化动作。 6. 执行构造器。你将会在"复用"这一章看到,这可能会牵涉到很多动作,尤其当涉及继承的时候。