思考一个问题:关于下图程序第一次的运行结果。
package com.zq.chat.client;
class Singleton{
private static Singleton singleton = new Singleton();
public static int counter1;
public static int counter2 = 0;
private Singleton(){
counter1++;
counter2++;
}
public static Singleton getInstance(){
return singleton;
}
}
public class MyTest {
public static void main(String[] args) {
Singleton singleton = Singleton.getInstance();
System.out.println("counter1 = " + singleton.counter1);
System.out.println("counter2 = " + singleton.counter2);
}
}
很多人会给出
counter1 = 1
counter2 = 1
但实际是第一次运行的结果为:
这里就涉及到关于类加载的问题。
首先我写给出一些基础点:
类的初始化:
主动使用(六种)
- 创建类的实例
- 访问某个类或接口的静态变量,或对该静态变量赋值
- 调用类的静态方法
- 发射
- 初始化一个类的子类
- Java虚拟机启动时被表明启动类的类。
除了以上六种情况,其他的使用Java类的方式都被看作是对类的被动使用,都不会导致类的初始化。
类的加载:
有两种类型的类加载器
-Java虚拟机自带的加载器
-根类加载器(Bootstrap)
-扩展类类加载器(Extension)
-系统加载器(System)
-用户自定义加载器
-java.lang.ClassLoader的子类
-用户可以定制类的加载方式
类的加载指的是将类的.class文件中的二进制数据读入到内存当中,将其存放到运行时数据区方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区的数据结构。
*加载.class文件方式
- 从本地系统中直接加载(最多的自己写自己用)
- 通过网络下载.class文件
-
从zip,jar等归档文件中加载.class文件 (添加进的第三方)
- 从专有数据库中提取.class文件
- 将java源文件动态编译为.class文件
类的验证内容
- 类文件的结构检查
- 语义检查
- 字节码验证
- 二进制兼容性的验证
类的准备
类准备阶段,Java虚拟机为类的静态变量分配内存,并设置默认的初始化值。例如对某个带静态int与long变量的类,准备阶段,将为Int类型静态变量分配4个字节,并赋予默认值为0,为long类型的静态变量分配8个字节的内存空间,并赋予默认值为0.(
Java
在任意操作系统上都是如此,因为
Java
是运行在jvm上而不是运行在操作系统上)
类的解析
在解析阶段,
Java
虚拟机会把类的二进制数据中的符号引用替换为直接引用。
看完了上面的基础点,我们回到这里的例子。数据在初始化阶段, Java虚拟机启动时被表明启动类的类是MyTest 之后。 最先调用Singleton类的静态方法 加载此类 ->给所有静态变量赋值初始化 :
//括号后为每个变量的初始化值
private static Singleton singleton;(null)
public static int counter1;(0)
public static int counter2;(0)
此处注意:*是赋予初始化的值,而不是赋值!!。
之后才是代码从上到下执行:
private static Singleton singleton = new Singleton();//1 生成实例
public static int counter1;//4
public static int counter2 = 0;//5
private Singleton(){
counter1++;//2 生成实例运行私有构造器
counter2++;//3
}
因此一幕了然,最后的输出为
counter1 = 1
counter2 = 0
虽然此处单纯的从语法上可以直接得出答案,但是连接上jvm的知识,可以更好的入门,联系jvm的知识与基础的开发关系。