jvm之类加载器(1)

gg1993 8年前

来自: http://www.cnblogs.com/think-in-java/p/5178383.html

首先我们先看一个示例程序:

package com.tfdd.test;    /**   * @desc 类加载校验   * @author chenqm    * @date 2016年2月2日   */  class Singleton{      private static Singleton singleton = new Singleton();      public static int count1 ;      public static int count2 = 0;            private Singleton(){          count1++;          count2++;      }        public static Singleton getInstance(){                    return singleton;      }        }  public class SingletonTest {      public static void main(String[] args) {          Singleton singleton = Singleton.getInstance();          System.out.println(singleton.count1);          System.out.println(singleton.count2);      }  }

猜猜输出的结果是什么?据说80%的java程序猿都会犯的错误!

10

就是这样一个结果,我们先不说为什么。接着讲我们的类加载器.

类的加载大致分为三个部分:加载,连接,初始化。

加载 :查找并加载类的二进制数据

连接 :1.验证(确保被加载类的准确性) 2.准备(为类的静态变量分配内存,并将其初始化为默认值) 3.解析(将类中的符号引用转化为直接引用)

初始化 :为类的静态变量赋予正确的初始值(即赋上我们给出的值)

然后我们再看看java程序对类的使用方式:主动使用和被动使用。请注意下面我说的这句话

所有的JAVA虚拟机实现必须在每个类或接口被JAVA程序“首次主动使用”时才初始化他们。

那么什么叫主动使用?基本上分为6种情况:

--创建类的实例

--访问某个类或接口的静态变量,或者对该静态变量赋值

--调用类的静态方法

--反射

--初始化一个类的子类

--java虚拟机启动时被标明为启动类的类

了解了这些知识之后,我现在来回答之前的问题,

Singleton.getInstance(); 调用了类的静态方法,符合首次主动使用该类的情况!那么我们进入初始化阶段。  初始化已经属于类加载的第三步了,在第二步的连接的准备部分,已经赋过一次默认值了。所以应该是这样一个过程:  1.singleton = null;count1=0;count1=0 2.singleton = new Singleton(); 此时执行构造函数,结果为count0=1;count1=1 3.count0没有被我们赋值跳过初始化,count1赋值为0所以结果为count0=1;count1=0 (*注意) 类的静态变量赋值的顺序是按照代码的书写的顺序执行的。  修改代码如下:
package com.tfdd.test;    /**   * @desc 类加载校验   * @author chenqm    * @date 2016年2月2日   */  class Singleton{      public static int count1 ;      public static int count2 = 0;      private static Singleton singleton = new Singleton();        private Singleton(){          count1++;          count2++;      }        public static Singleton getInstance(){                    return singleton;      }        }  public class SingletonTest {      public static void main(String[] args) {          Singleton singleton = Singleton.getInstance();          System.out.println(singleton.count1);          System.out.println(singleton.count2);      }  }

输出结果为

1

1

证明赋值的顺序是根据代码书写的先后顺序执行的!