`

【J2SE】详解java类的生命周期

    博客分类:
  • J2SE
 
阅读更多

本文转自:http://www.2cto.com/kf/201204/129386.html

引言
         最近有位细心的朋友在阅读笔者的文章时,对java类的生命周期问题有一些疑惑,笔者打开百度搜了一下相关的问题,看到网上的资料很少有把这个问题讲明白的,主要是因为目前国内java方面的教材大多只是告诉你“怎样做”,但至于“为什么这样做”却不多说,所以造成大家在基础和原理方面的知识比较匮乏,所以笔者今天就斗胆来讲一下这个问题,权当抛砖引玉,希望对在这个问题上有疑惑的朋友有所帮助,文中有说的不对的地方,也希望各路高手前来指正。
         首先来了解一下jvm(java虚拟机)中的几个比较重要的内存区域,这几个区域在java类的生命周期中扮演着比较重要的角色:
方法区:在java的虚拟机中有一块专门用来存放已经加载的类信息、常量、静态变量以及方法代码的内存区域,叫做方法区。
常量池:常量池是方法区的一部分,主要用来存放常量和类中的符号引用等信息。
堆区:用于存放类的对象实例。
栈区:也叫java虚拟机栈,是由一个一个的栈帧组成的后进先出的栈式结构,栈桢中存放方法运行时产生的局部变量、方法出口等信息。当调用一个方法时,虚拟机栈中就会创建一个栈帧存放这些数据,当方法调用完成时,栈帧消失,如果方法中调用了其他方法,则继续在栈顶创建新的栈桢。
         除了以上四个内存区域之外,jvm中的运行时内存区域还包括本地方法栈和程序计数器,这两个区域与java类的生命周期关系不是很大,在这里就不说了,感兴趣的朋友可以自己百度一下。
 
类的生命周期
         当我们编写一个java的源文件后,经过编译会生成一个后缀名为class的文件,这种文件叫做字节码文件,只有这种字节码文件才能够在java虚拟机中运行,java类的生命周期就是指一个class文件从加载到卸载的全过程。
         一个java类的完整的生命周期会经历加载、连接、初始化、使用、和卸载五个阶段,当然也有在加载或者连接之后没有被初始化就直接被使用的情况,如图所示:



下面我们就依次来说一说这五个阶段。


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

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

从网络中获取:比如10年前十分流行Applet。
根据一定的规则实时生成,比如设计模式中的动态代理模式,就是根据相应的类自动生成它的代理类。
从非class文件中获取,其实这与直接从class文件中获取的方式本质上是一样的,这些非class文件在jvm中运行之前会被转换为可被jvm所识别的字节码文件。
        对于加载的时机,各个虚拟机的做法并不一样,但是有一个原则,就是当jvm“预期”到一个类将要被使用时,就会在使用它之前对这个类进行加载。比如说,在一段代码中出现了一个类的名字,jvm在执行这段代码之前并不能确定这个类是否会被使用到,于是,有些jvm会在执行前就加载这个类,而有些则在真正需要用的时候才会去加载它,这取决于具体的jvm实现。我们常用的hotspot虚拟机是采用的后者,就是说当真正用到一个类的时候才对它进行加载。

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



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

验证:当一个类被加载之后,必须要验证一下这个类是否合法,比如这个类是不是符合字节码的格式、变量与方法是不是有重复、数据类型是不是有效、继承与实现是否合乎标准等等。总之,这个阶段的目的就是保证加载的类是能够被jvm所运行。
准备:准备阶段的工作就是为类的静态变量分配内存并设为jvm默认的初值,对于非静态的变量,则不会为它们分配内存。有一点需要注意,这时候,静态变量的初值为jvm默认的初值,而不是我们在程序中设定的初值。jvm默认的初值是这样的:
基本类型(int、long、short、char、byte、boolean、float、double)的默认值为0。
引用类型的默认值为null。
常量的默认值为我们程序中设定的值,比如我们在程序中定义final static int a = 100,则准备阶段中a的初值就是100。
  解析:这一阶段的任务就是把常量池中的符号引用转换为直接引用。那么什么是符号引用,什么又是直接引用呢?我们来举个例子:我们要找一个人,我们现有的信息是这个人的身份证号是1234567890。只有这个信息我们显然找不到这个人,但是通过公安局的身份系统,我们输入1234567890这个号之后,就会得到它的全部信息:比如安徽省黄山市余暇村18号张三,通过这个信息我们就能找到这个人了。这里,123456790就好比是一个符号引用,而安徽省黄山市余暇村18号张三就是直接引用。在内存中也是一样,比如我们要在内存中找一个类里面的一个叫做show的方法,显然是找不到。但是在解析阶段,jvm就会把show这个名字转换为指向方法区的的一块内存地址,比如c17164,通过c17164就可以找到show这个方法具体分配在内存的哪一个区域了。这里show就是符号引用,而c17164就是直接引用。在解析阶段,jvm会将所有的类或接口名、字段名、方法名转换为具体的内存地址。
         连接阶段完成之后会根据使用的情况(直接引用还是被动引用)来选择是否对类进行初始化。



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

通过new关键字实例化对象、读取或设置类的静态变量、调用类的静态方法。
通过反射方式执行以上三种行为。
初始化子类的时候,会触发父类的初始化。
作为程序入口直接运行时(也就是直接调用main方法)。
         除了以上四种情况,其他使用类的方式叫做被动引用,而被动引用不会触发类的初始化。请看主动引用的示例代码:

[java]  import java.lang.reflect.Field;
import java.lang.reflect.Method;
 
class InitClass{
     static {
         System.out.println("初始化InitClass");
     }
     public static String a = null;
     public static void method(){}
}
 
class SubInitClass extends InitClass{}
 
public class Test1 {
 
     /**
      * 主动引用引起类的初始化的第四种情况就是运行Test1的main方法时
      * 导致Test1初始化,这一点很好理解,就不特别演示了。
      * 本代码演示了前三种情况,以下代码都会引起InitClass的初始化,
      * 但由于初始化只会进行一次,运行时请将注解消掉,依次运行查看结果。
      * @param args
      * @throws Exception
      */
     public static void main(String[] args) throws Exception{
     //  主动引用引起类的初始化一: new对象、读取或设置类的静态变量、调用类的静态方法。 
     //  new InitClass(); 
     //  InitClass.a = ""; 
     //  String a = InitClass.a; 
     //  InitClass.method(); 
         
     //  主动引用引起类的初始化二:通过反射实例化对象、读取或设置类的静态变量、调用类的静态方法。 
     //  Class cls = InitClass.class; 
     //  cls.newInstance(); 
         
     //  Field f = cls.getDeclaredField("a"); 
     //  f.get(null); 
     //  f.set(null, "s"); 
     
     //  Method md = cls.getDeclaredMethod("method"); 
     //  md.invoke(null, null); 
             
     //  主动引用引起类的初始化三:实例化子类,引起父类初始化。 
     //  new SubInitClass(); 
 
     }
}
import java.lang.reflect.Field;
import java.lang.reflect.Method;

class InitClass{
  static {
   System.out.println("初始化InitClass");
  }
  public static String a = null;
  public static void method(){}
}

class SubInitClass extends InitClass{}

public class Test1 {

/**
   * 主动引用引起类的初始化的第四种情况就是运行Test1的main方法时
   * 导致Test1初始化,这一点很好理解,就不特别演示了。
   * 本代码演示了前三种情况,以下代码都会引起InitClass的初始化,
   * 但由于初始化只会进行一次,运行时请将注解消掉,依次运行查看结果。
   * @param args
   * @throws Exception
   */
  public static void main(String[] args) throws Exception{
  //  主动引用引起类的初始化一: new对象、读取或设置类的静态变量、调用类的静态方法。
  // new InitClass();
  // InitClass.a = "";
  // String a = InitClass.a;
  // InitClass.method();
  
  //  主动引用引起类的初始化二:通过反射实例化对象、读取或设置类的静态变量、调用类的静态方法。
  // Class cls = InitClass.class;
  // cls.newInstance();
  
  // Field f = cls.getDeclaredField("a");
  // f.get(null);
  // f.set(null, "s");
 
  // Method md = cls.getDeclaredMethod("method");
  // md.invoke(null, null);
   
  //  主动引用引起类的初始化三:实例化子类,引起父类初始化。
  // new SubInitClass();

}
}
 
         上面的程序演示了主动引用触发类的初始化的四种情况。



        类的初始化过程是这样的:按照顺序自上而下运行类中的变量赋值语句和静态语句,如果有父类,则首先按照顺序运行父类中的变量赋值语句和静态语句。先看一个例子,首先建两个类用来显示赋值操作:

[java] public class Field1{
     public Field1(){
         System.out.println("Field1构造方法");
     }
}
public class Field2{
     public Field2(){
         System.out.println("Field2构造方法");
     }
}
public class Field1{
  public Field1(){
   System.out.println("Field1构造方法");
  }
}
public class Field2{
  public Field2(){
   System.out.println("Field2构造方法");
  }
}
 
下面是演示初始化顺序的代码:

[java] class InitClass2{
     static{
         System.out.println("运行父类静态代码");
     }
     public static Field1 f1 = new Field1();
     public static Field1 f2; 
}
 
class SubInitClass2 extends InitClass2{
     static{
         System.out.println("运行子类静态代码");
     }
     public static Field2 f2 = new Field2();
}
 
public class Test2 {
     public static void main(String[] args) throws ClassNotFoundException{
         new SubInitClass2();
     }
}
class InitClass2{
  static{
   System.out.println("运行父类静态代码");
  }
  public static Field1 f1 = new Field1();
  public static Field1 f2;
}

class SubInitClass2 extends InitClass2{
  static{
   System.out.println("运行子类静态代码");
  }
  public static Field2 f2 = new Field2();
}

public class Test2 {
  public static void main(String[] args) throws ClassNotFoundException{
   new SubInitClass2();
  }
}
         上面的代码中,初始化的顺序是:第03行,第05行,第11行,第13行。第04行是声明操作,没有赋值,所以不会被运行。而下面的代码:

[java] class InitClass2{
     public static Field1 f1 = new Field1();
     public static Field1 f2;
     static{
         System.out.println("运行父类静态代码");
     }
}
 
class SubInitClass2 extends InitClass2{
     public static Field2 f2 = new Field2();
     static{
         System.out.println("运行子类静态代码");
     }
}
 
public class Test2 {
     public static void main(String[] args) throws ClassNotFoundException{
         new SubInitClass2();
     }
}
class InitClass2{
  public static Field1 f1 = new Field1();
  public static Field1 f2;
  static{
   System.out.println("运行父类静态代码");
  }
}

class SubInitClass2 extends InitClass2{
  public static Field2 f2 = new Field2();
  static{
   System.out.println("运行子类静态代码");
  }
}

public class Test2 {
  public static void main(String[] args) throws ClassNotFoundException{
   new SubInitClass2();
  }
}
 
         初始化顺序为:第02行、第05行、第10行、第12行,各位可以运行程序查看结果。

       在类的初始化阶段,只会初始化与类相关的赋值语句和静态语句,也就是有static关键字修饰的信息,没有static修饰的赋值语句和静态语句在实例化对象的时候才会运行。



使用
        类的使用包括主动引用和被动引用,主动引用在初始化的章节中已经说过了,下面我们主要来说一下被动引用:

引用父类的静态字段,只会引起父类的初始化,而不会引起子类的初始化。
定义类数组,不会引起类的初始化。
引用类的常量,不会引起类的初始化。
请看一下被动引用的示例代码:

[java] class InitClass{
     static {
         System.out.println("初始化InitClass");
     }
     public static String a = null;
     public final static String b = "b";
     public static void method(){}
}
 
class SubInitClass extends InitClass{
     static {
         System.out.println("初始化SubInitClass");
     }
}
 
public class Test4 {
 
     public static void main(String[] args) throws Exception{
     //  String a = SubInitClass.a;// 引用父类的静态字段,只会引起父类初始化,而不会引起子类的初始化 
     //  String b = InitClass.b;// 使用类的常量不会引起类的初始化 
         SubInitClass[] sc = new SubInitClass[10];// 定义类数组不会引起类的初始化 
     }
}
class InitClass{
  static {
   System.out.println("初始化InitClass");
  }
  public static String a = null;
  public final static String b = "b";
  public static void method(){}
}

class SubInitClass extends InitClass{
  static {
   System.out.println("初始化SubInitClass");
  }
}

public class Test4 {

public static void main(String[] args) throws Exception{
  // String a = SubInitClass.a;// 引用父类的静态字段,只会引起父类初始化,而不会引起子类的初始化
  // String b = InitClass.b;// 使用类的常量不会引起类的初始化
   SubInitClass[] sc = new SubInitClass[10];// 定义类数组不会引起类的初始化
  }
}
 

        最后总结一下使用阶段:使用阶段包括主动引用和被动引用,主动饮用会引起类的初始化,而被动引用不会引起类的初始化。

        当使用阶段完成之后,java类就进入了卸载阶段。



卸载
        关于类的卸载,笔者在单例模式讨论篇:单例模式与垃圾回收一文中有过描述,在类使用完之后,如果有下面的情况,类就会被卸载:

该类所有的实例都已经被回收,也就是java堆中不存在该类的任何实例。
加载该类的ClassLoader已经被回收。
该类对应的java.lang.Class对象没有任何地方被引用,无法在任何地方通过反射访问该类的方法。
         如果以上三个条件全部满足,jvm就会在方法区垃圾回收的时候对类进行卸载,类的卸载过程其实就是在方法区中清空类信息,java类的整个生命周期就结束了。



总结
         做java的朋友对于对象的生命周期可能都比较熟悉,对象基本上都是在jvm的堆区中创建,在创建对象之前,会触发类加载(加载、连接、初始化),当类初始化完成后,根据类信息在堆区中实例化类对象,初始化非静态变量、非静态代码以及默认构造方法,当对象使用完之后会在合适的时候被jvm垃圾收集器回收。读完本文后我们知道,对象的生命周期只是类的生命周期中使用阶段的主动引用的一种情况(即实例化类对象)。而类的整个生命周期则要比对象的生命周期长的多。
  • 大小: 13.1 KB
分享到:
评论

相关推荐

    j2se基础课程资源

    学习线程的创建、同步、生命周期管理,以及线程安全问题的处理,对于开发高并发应用至关重要。 这个资源包为Java初学者提供了一个完整的学习路径,从基本语法到面向对象编程,再到高级特性如多线程和数据库编程,...

    J2SE面试题 经典

    ### J2SE经典面试知识点详解 #### 面向对象的四大特征 在J2SE面试中,理解面向对象编程的四个核心特征至关重要,这包括**抽象**、**继承**、**封装**以及**多态性**。 1. **抽象**:抽象是面向对象编程的基础之一...

    java2编程详解(special_edition_using_java)2.rar

    6. **多线程**:包括线程的创建、同步、生命周期管理,以及并发工具类的使用。 7. **网络编程**:讲解Socket编程,以及ServerSocket类的应用,实现客户端和服务器之间的通信。 8. **Java Applet**:介绍如何编写和...

    java面试题题库详解

    - **Servlet 生命周期:** 初始化、服务、销毁三个阶段。 - **配置方式:** 在 web.xml 文件中配置 Servlet。 - **实例化与重用:** Servlet 的实例化时机和复用机制。 #### 14. Thread的destroy()方法 **知识点...

    Java发展史_&_Java9、10新特性

    Java 10对垃圾回收机制进行了优化,提高了效率,特别是针对短生命周期对象的回收。 - **JShell命令行工具** JShell是一个交互式的Java shell工具,允许开发者在命令行中即时编写和测试代码片段,这对于快速原型...

    java发展历程1.zip

    Java发展历程详解 Java,一种广泛使用的面向对象的编程语言,由James Gosling在1991年于Sun Microsystems公司创立,至今已有三十多年的历史。Java的设计理念是“一次编写,到处运行”(Write Once, Run Anywhere)...

    JAVA考试题.pdf

    本题涉及到Java的基础知识,主要包括Java的运行平台、开发工具、类路径、面向对象特性、虚拟机执行特点、语法规范以及程序生命周期。 1. **Java运行平台** - Java提供了三个主要版本:J2SE(Java Standard Edition...

    java复习题300道

    3. **程序类型与生命周期** - Java有两种程序形式:Application(应用程序)和Applet(小程序)。前者在main()方法启动,后者在init()方法启动。 - 绘图程序通常使用paint(Graphics g)方法进行图形绘制。 4. **...

    (超赞)JAVA精华之--深入JAVA API

    - Java 线程模型、线程生命周期、线程同步等。 **1.7 Java 5.0多线程编程** - **高级主题** - 如并发工具类、锁机制等。 **1.8 Java Socket编程** - **网络编程基础** - 使用 `Socket` 和 `ServerSocket` 类...

    JAVA 基础JSE.pdf

    2. **一种软件开发平台**:Java提供了完整的开发环境,支持从需求分析到维护的所有软件开发生命周期阶段。这包括但不限于需求分析、概要设计、详细设计、编码、测试以及维护等步骤。 - **需求分析**:明确项目的...

    Java程序设计习题[参考].pdf

    此外,Java没有析构方法(如C++中的`delete`),这意味着对象的生命周期管理和内存释放完全由Java的垃圾回收机制自动处理。 - **结构运行时内存释放**:由于Java不使用手动内存管理,当一个对象不再被引用时,Java...

    Java程序设计实验指导书(答案)[归类].pdf

    3. **理解Java程序的生命周期**:了解Java源代码、字节码文件,熟悉Java程序的编辑、编译和运行过程。 **实验任务** - **下载与安装J2SE**:从网络上下载或直接从CD-ROM安装J2SE开发工具。 - **编写简单程序**:...

    java学习笔记

    - **多线程编程**: 处理并发问题的技术,如线程的创建、同步和生命周期管理。 #### 五、Java EE内容 Java EE是针对企业级应用的一套完整的解决方案,其内容包括但不限于: - **基础**: - 面向对象编程 - 数据库...

    如何从零开始学习JavaEE?

    1. 学习Servlet和JSP的基础知识,了解Servlet的生命周期和JSP的编译过程。 2. 学习Struts、Hibernate、Spring等框架的基础知识,了解MVC模型和Java@Web开发技术。 3. 学习Servlet和JSP的高级知识,掌握Servlet和JSP...

    java网络编程基础习题+部分答案

    它是在 Applet 生命周期中的最后一个被调用的方法。 **8. 面向大型企业级应用平台** - **选项分析:** - A)J2EE:面向大型企业级应用。 - B)J2ME:面向移动设备。 - C)J2SE:标准版 Java 平台。 - D)J2DE...

    MLDN笔记(含J2SE,Web和框架部分)

    - **Servlet与JSP**:学习Servlet生命周期,理解JSP页面转换为Servlet的过程。 - **MVC设计模式**:理解Model-View-Controller模式在Web开发中的应用。 - **JSTL与EL表达式**:掌握JSP标准标签库(JSTL)和表达式...

    java实用教程

    通过上述内容,我们可以看到Java不仅仅是一种编程语言,它还是一种全面的软件开发平台,为开发者提供了从编写到部署整个生命周期所需的所有工具和技术。无论是大型企业还是小型创业公司,Java都能提供强大的支持,...

    ibm_java(ibm课程系列)

    - **项目管理**:支持项目生命周期管理,包括版本控制、构建配置等。 - **代码编辑器**:具备强大的编辑功能,如代码自动完成、语法高亮等。 - **调试工具**:内置调试器,支持断点设置、变量查看等功能。 - **部署...

    IBM JAVA培训计划

    - Servlet生命周期 - MVC2模式 - Tomcat服务器配置 ##### 6. JavaScript - 基础语法 - 对象编程 - 事件处理 ##### 7. Ajax - XMLHttpRequest对象 - JSON/XML数据交换 - Ajax框架 ##### 8. UML与项目管理 - UML...

Global site tag (gtag.js) - Google Analytics