详解Java类的生命周期

上一篇 / 下一篇  2012-05-08 09:48:35 / 个人分类:Java

引言

j Z&RZxsm0  最近有位细心的朋友在阅读笔者的文章时,对java类的生命周期问题有一些疑惑,笔者打开百度搜 了一下相关的问题,看到网上的资料很少有把这个问题讲明白的,主要是因为目前国内java方面的教材大多只是告诉你“怎样做”,但至于“为什么这样做”却 不多说,所以造成大家在基础和原理方面的知识比较匮乏,所以笔者今天就斗胆来讲一下这个问题,权当抛砖引玉,希望对在这个问题上有疑惑的朋友有所帮助,文 中有说的不对的地方,也希望各路高手前来指正。51Testing软件测试网Q'H$_.{P }2X4O

)Y3L!?8v%p0  首先来了解一下jvm(java虚拟机)中的几个比较重要的内存区域,这几个区域在java类的生命周期中扮演着比较重要的角色:

BaR`l|P6eR [051Testing软件测试网z6R!v?G`:E

  ● 方法区:在java的虚拟机中有一块专门用来存放已经加载的类信息、常量、静态变量以及方法代码的内存区域,叫做方法区。

'Z-t!Y J)~~051Testing软件测试网!Uw \ ][{hJ

  ● 常量池:常量池是方法区的一部分,主要用来存放常量和类中的符号引用等信息。

n t s}+g7D t051Testing软件测试网 a)s*jGuW d&dL

  ● 堆区:用于存放类的对象实例。51Testing软件测试网 c4@&MG"SL^@

i/K+x\E0za0   ● 栈区:也叫java虚拟机栈,是由一个一个的栈帧组成的后进先出的栈式结构,栈桢中存放方法运行时产生的局部变量、方法出口等信息。当调用一个方法时,虚 拟机栈中就会创建一个栈帧存放这些数据,当方法调用完成时,栈帧消失,如果方法中调用了其他方法,则继续在栈顶创建新的栈桢。51Testing软件测试网o7EQet

51Testing软件测试网5ZzE6b_*P'_U

  除了以上四个内存区域之外,jvm中的运行时内存区域还包括本地方法栈和程序计数器,这两个区域与java类的生命周期关系不是很大,在这里就不说了,感兴趣的朋友可以自己百度一下。

/Smw?(M2l051Testing软件测试网KS].[c#x `,x

  类的生命周期

}o(F&L:i)EH051Testing软件测试网 FA|'`;N1X

  当我们编写一个java的源文件后,经过编译会生成一个后缀名为class的文件,这种文件叫做字节码文件,只有这种字节码文件才能够在java虚拟机中运行,java类的生命周期就是指一个class文件从加载到卸载的全过程。

x;m]dIp*oM0

0?gll0Ht;x0  一个java类的完整的生命周期会经历加载、连接、初始化、使用、和卸载五个阶段,当然也有在加载或者连接之后没有被初始化就直接被使用的情况,如图所示:51Testing软件测试网(z:~6uCW za6]8?9j

51Testing软件测试网/@ dg&L;z,? K}4r0v&i[

  下面我们就依次来说一说这五个阶段。51Testing软件测试网F1HG T)|3K

  加载

#g7Mp0G be0

  在java中,我们经常会接触到一个词——类加载,它和这里的加载并不是一回事,通常我们说类加载指的是类的生命周期中加载、连接、初始化三个阶段。在加载阶段,java虚拟机会做什么工作呢?其实很简单,就是找到需要加载的类并把类的信息加载到jvm的方法区中,然后在堆区中实例化一个java.lang.Class对象,作为方法区中这个类的信息的入口。

g#\.@:WK mJ;V0

  类的加载方式比较灵活,我们最常用的加载方式有两种,一种是根据类的全路径名找到相应的class文件,然后从class文件中读取文件内容;另一种是从jar文件中读取。另外,还有下面几种方式也比较常用:51Testing软件测试网]Dr ]U+e#e

  ● 从网络中获取:比如10年前十分流行Applet。

'Zf go)Ba Gp3B;P0

  ● 根据一定的规则实时生成,比如设计模式中的动态代理模式,就是根据相应的类自动生成它的代理类。

~I4F`}*kD5w N0

  ● 从非class文件中获取,其实这与直接从class文件中获取的方式本质上是一样的,这些非class文件在jvm中运行之前会被转换为可被jvm所识别的字节码文件。51Testing软件测试网Y[il_+{

   对于加载的时机,各个虚拟机的做法并不一样,但是有一个原则,就是当jvm“预期”到一个类将要被使用时,就会在使用它之前对这个类进行加载。比如说, 在一段代码中出现了一个类的名字,jvm在执行这段代码之前并不能确定这个类是否会被使用到,于是,有些jvm会在执行前就加载这个类,而有些则在真正需 要用的时候才会去加载它,这取决于具体的jvm实现。我们常用的hotspot虚拟机是采用的后者,就是说当真正用到一个类的时候才对它进行加载。51Testing软件测试网;t ^$q7B0Sw yf'\4R

   加载阶段是类的生命周期中的第一个阶段,加载阶段之后,是连接阶段。有一点需要注意,就是有时连接阶段并不会等加载阶段完全完成之后才开始,而是交叉进 行,可能一个类只加载了一部分之后,连接阶段就已经开始了。但是这两个阶段总的开始时间和完成时间总是固定的:加载阶段总是在连接阶段之前开始,连接阶段 总是在加载阶段完成之后完成。51Testing软件测试网4p3Z'ZYAsx'~4g

连接

O/` g8{1VHkrm0  连接阶段比较复杂,一般会跟加载阶段和初始化阶段交叉进行,这个阶段的主要任务就是做一些加载后的验证工作以及一些初始化前的准备工作,可以细分为三个步骤:验证、准备和解析。

5gSuyp Q0

bST E LD-H'p0  1、验证:当一个类被加载之后,必须要验证一下这个类是否合法,比如这个类是不是符合字节码的格式、变量与方法是不是有重复、数据类型是不是有效、继承与实现是否合乎标准等等。总之,这个阶段的目的就是保证加载的类是能够被jvm所运行。

a2M#L|9FZ1M/Jy0

[3d(o'f4]bg2F0  2、准备:准备阶段的工作就是为类的静态变量分配内存并设为jvm默认的初值,对于非静态的变量,则不会为它们分配内存。有一点需要注意,这时候,静态变量的初值为jvm默认的初值,而不是我们在程序中设定的初值。jvm默认的初值是这样的:

@(N!M`3ZK Si0

B-u0W#LQxz0s`6R0  ● 基本类型(int、long、short、char、byte、boolean、float、double)的默认值为0。

*Q+GMEq"T&p0

B'SaL NIp0  ● 引用类型的默认值为null。51Testing软件测试网'y6z|DB#G%e

51Testing软件测试网 { H K~i5I:Q

  ● 常量的默认值为我们程序中设定的值,比如我们在程序中定义final static int a = 100,则准备阶段中a的初值就是100。51Testing软件测试网2W*D/Z+s}w

51Testing软件测试网0?-s^wsE,\eLU

  3、解析:这一阶段的任务就是把常量池中的符号引用转换为直接引用。那么什么是符号引用,什么又是直接引用呢?我们来举个例子:我们要找一个 人,我们现有的信息是这个人的身份证号是1234567890。只有这个信息我们显然找不到这个人,但是通过公安局的身份系统,我们输入 1234567890这个号之后,就会得到它的全部信息:比如安徽省黄山市余暇村18号张三,通过这个信息我们就能找到这个人了。这 里,123456790就好比是一个符号引用,而安徽省黄山市余暇村18号张三就是直接引用。在内存中也是一样,比如我们要在内存中找一个类里面的一个叫 做show的方法,显然是找不到。但是在解析阶段,jvm就会把show这个名字转换为指向方法区的的一块内存地址,比如c17164,通过c17164 就可以找到show这个方法具体分配在内存的哪一个区域了。这里show就是符号引用,而c17164就是直接引用。在解析阶段,jvm会将所有的类或接 口名、字段名、方法名转换为具体的内存地址。

R'Ud k s&mQ!{1d051Testing软件测试网'imX!w!W#n5z

  4、连接阶段完成之后会根据使用的情况(直接引用还是被动引用)来选择是否对类进行初始化。51Testing软件测试网'kJej5L%O)O)L4{

51Testing软件测试网qD,O+M\GOW,v

  初始化51Testing软件测试网g4Z!x[(s3mp1{'lj

51Testing软件测试网{kEt q!Y!z

  如果一个类被直接引用,就会触发类的初始化。在java中,直接引用的情况有:

Kp~8ZE051Testing软件测试网+cv1Gx/u%K'H4H7tGL

  通过new关键字实例化对象、读取或设置类的静态变量、调用类的静态方法。51Testing软件测试网+jQ{rYd Jn j ?#SvA

51Testing软件测试网C8^&SjLXR|

  通过反射方式执行以上三种行为。

5SF8J9oN;I051Testing软件测试网0u,rJ"RL"w.pB

  初始化子类的时候,会触发父类的初始化。51Testing软件测试网n5V3tS*v3h

(htw;K^0  作为程序入口直接运行时(也就是直接调用main方法)。51Testing软件测试网.P;e,[ AdVz_

51Testing软件测试网0J'LW9M9R:AE

  除了以上四种情况,其他使用类的方式叫做被动引用,而被动引用不会触发类的初始化。请看主动引用的示例代码:51Testing软件测试网 Bz H-Rs.{yy@&i

F.c{8[Ey0

o(i(b*wf4|:ll0
  1. import java.lang.reflect.Field;  
  2. import java.lang.reflect.Method;  
  3.  
  4. class InitClass{  
  5.     static {  
  6.         System.out.println("初始化InitClass");  
  7.     }  
  8.     public static String a = null;  
  9.     public static void method(){}  
  10. }  
  11.  
  12. class SubInitClass extends InitClass{}  
  13.  
  14. public class Test1 {  
  15.  
  16.     /** 
  17.      * 主动引用引起类的初始化的第四种情况就是运行Test1的main方法时 
  18.      * 导致Test1初始化,这一点很好理解,就不特别演示了。 
  19.      * 本代码演示了前三种情况,以下代码都会引起InitClass的初始化, 
  20.      * 但由于初始化只会进行一次,运行时请将注解去掉,依次运行查看结果。 
  21.      * @param args 
  22.      * @throws Exception 
  23.      */ 
  24.     public static void main(String[] args) throws Exception{  
  25.     //  主动引用引起类的初始化一: new对象、读取或设置类的静态变量、调用类的静态方法。 
  26.     //  new InitClass(); 
  27.     //  InitClass.a = ""; 
  28.     //  String a = InitClass.a; 
  29.     //  InitClass.method(); 
  30.           
  31.     //  主动引用引起类的初始化二:通过反射实例化对象、读取或设置类的静态变量、调用类的静态方法。 
  32.     //  Class cls = InitClass.class; 
  33.     //  cls.newInstance(); 
  34.           
  35.     //  Field f = cls.getDeclaredField("a"); 
  36.     //  f.get(null); 
  37.     //  f.set(null, "s"); 
  38.       
  39.     //  Method md = cls.getDeclaredMethod("method"); 
  40.     //  md.invoke(null, null); 
  41.               
  42.     //  主动引用引起类的初始化三:实例化子类,引起父类初始化。 
  43.     //  new SubInitClass(); 
  44.  
  45.     }  
  46. }
51Testing软件测试网oc |:s3v%m

  上面的程序演示了主动引用触发类的初始化的四种情况。51Testing软件测试网q ]/t(oN


TAG:

 

评分:0

我来说两句

Open Toolbar