一般不假思索的结论就是,a=1,b=1。给出的原因是:a、b 都是静态变量,在构造函数
调用的时候已经对 a 和 b 都加 1 了。答案就都是 1。但是运行完后答案却是 a=1,b=0。
下面我们将代码稍微变一下
public static Singleton singleton = new Singleton();
public static int a;
public static int b = 0;
的代码部分替换成
public static int a;
public static int b = 0;
public static Singleton singleton = new Singleton();
效果就是刚才预期的 a=1,b=1。
为什么呢,这 3 句无非就是静态变量的声明、初始化,值的变化和声明的顺序还有关
系吗?Java 不是面向对象的吗?怎么和结构化的语言似地,顺序还有关系。这个就是
和 Java 虚拟机 JVM 加载类的原理有着直接的关系。
3. 类在 JVM 中的工作原理
要想使用一个 Java 类为自己工作,必须经过以下几个过程
1):类加载 load:从字节码二进制文件——.class 文件将类加载到内存,从而达到类
的从硬盘上到内存上的一个迁移,所有的程序必须加载到内存才能工作。将内存中的
class 放到运行时数据区的方法区内,之后在堆区建立一个 java.lang.Class 对象,用来封
装方法区的数据结构。这个时候就体现出了万事万物皆对象了,干什么事情都得有个
对象。就是到了最底层究竟是鸡生蛋,还是蛋生鸡呢?类加载的最终产物就是堆中的
一个 java.lang.Class 对象。
2):连接:连接又分为以下小步骤
验证:出于安全性的考虑,验证内存中的字节码是否符合 JVM 的规范,类的结构规范、
语义检查、字节码操作是否合法、这个是为了防止用户自己建立一个非法的 XX.class 文
件就进行工作了,或者是 JVM 版本冲突的问题,比如在 JDK6 下面编译通过的 class(其
中包含注解特性的类),是不能在 JDK1.4 的 JVM 下运行的。
准备:将类的静态变量进行分配内存空间、初始化默认值。(对象还没生成呢,所以
这个时候没有实例变量什么事情)
解析:把类的符号引用转为直接引用(保留)
3):类的初始化: 将类的静态变量赋予正确的初始值,这个初始值是开发者自己定
义时赋予的初始值,而不是默认值。
4. 类的主动使用与被动使用
以下是视为主动使用一个类,其他情况均视为被动使用!
1):初学者最为常用的 new 一个类的实例对象(声明不叫主动使用)
2):对类的静态变量进行读取、赋值操作的。
3):直接调用类的静态方法。
4):反射调用一个类的方法。
5):初始化一个类的子类的时候,父类也相当于被程序主动调用了(如果调用子类的
静态变量是从父类继承过来并没有复写的,那么也就相当于只用到了父类的东东,和
子类无关,所以这个时候子类不需要进行类初始化)。
6):直接运行一个 main 函数入口的类。
所有的 JVM 实现(不同的厂商有不同的实现,有人就说 IBM 的实现比 Sun 的要好……)在
2 / 7