`
zhang.jianjun
  • 浏览: 20223 次
  • 性别: Icon_minigender_1
  • 来自: 青岛
最近访客 更多访客>>
社区版块
存档分类
最新评论
  • NGG: 个人建议: 正确的做法应该是把样式放到外部css文件中,使用 ...
    js日期选择器

【JAVA优化编程】内存管理之——(2)JVM中对象的生命周期

阅读更多

2  JVM中对象的生命周期

    在JVM运行空间中,对象的整个生命周期大致可以分为7个阶段:创建阶段(Creation)、应用阶段(Using)、不可视阶段 (Invisible)、不可到达阶段(Unreachable)、可收集阶段(Collected)、终结阶段(Finalized)、释放阶段 (Free)。上面的这7个阶段构成了JVM中对象的完整的生命周期。下面分别介绍这7个阶段。

 

2.1  创建阶段

    在对象创建阶段,系统要通过下面的步骤,完成对象的创建过程:

    (1) 为对象分配存储空间。

    (2) 开始构造对象。

    (3) 递归调用其超类的构造方法。

    (4) 进行对象实例初始化与变量初始化。

    (5) 执行构造方法体。

 

    上面的5个步骤中的第3步就是指递归地调用该类所扩展的所有父类的构造方法,一个Java类(除Object类外)至少有一个父类(Object),这个 规则既是强制的,也是隐式的。你可能已经注意到在创建一个Java类的时候,并没有显式地声明扩展(extends)一个Object父类。实际上,在 Java程序设计中,任何一个Java类都直接或间接的是Object类的子类。例如下面的代码:

public class A {
        ...
}
 

 

    这个声明等同于下面的声明:

public class A extends java.lang.Object {
    ...
}
 

 

    下面是在创建对象时的几个关键应用规则:

    (1) 避免在循环体中创建对象,即使该对象占用内存空间不大。

    (2) 尽量及时使对象符合垃圾回收标准。
    (3) 不要采用过深的继承层次。
    (4) 访问本地变量优于访问类中的变量。

    关于规则(1)避免在循环体中创建对象,即使该对象占用内存空间不大,需要提示一下,这种情况在我们的实际应用中经常遇到而且我们很容易犯类似的错误,例如下面的代码:

... ...
for (int i = 0; i < 10000; ++i) {
    Object obj = new Object();
    System.out.println("obj= " + obj);
}
... ...
 

 

    上面代码的书写方式相信对你来说不会陌生,也许在以前的应用开发中你也这样做过,尤其是在枚举一个Vector对象中的对象元素的操作中经常会这样书写,但这却违反了上述规则(1),因为这样会浪费较大的内存空间,正确的方法如下所示:

... ...
Object obj = null;
for (int i = 0; i < 10000; ++i) {
    obj = new Object();
    System.out.println("obj= " + obj);
}
... ...
 

 

    采用下面的编写方式,仅在内存中保存一份对该对象的引用,而不像上面的代码会在内存中产生大量的对象应用,浪费大量的内存空间,而且增大了系统做垃圾回收的负荷。因此在循环体中声明创建对象的编写方式应该尽量避免。

    另外,不要对一个对象初始化多次,这同样会带来较大的内存开销,降低系统性能,如:

public class A {
    private Hashtable table = new Hashtable();
    public A() {
        // 将Hashtable对象table初始化了两次
        table = new Hashtable();
    }
}
 

 

    正确的方式为:

public class B {
    private Hashtable table = new Hashtable();
    public B() {
    }
}
 

 

    不要小看这个差别,它却使应用软件的性能相差甚远,如图所示。

初始化对象多次所带来的性能差别

    看来在程序设计中也应该遵从“勿以恶小而为之”的古训,否则开发出来的应用也是个低效的应用,有时应用软件中的一个极小的失误,就会大幅度地降低整个系统的性能。

 

2.2  应用阶段

    在对象的应用阶段,对象具备下列特性:

  • 系统至少维护着对象的一个强引用(Strong Reference);
  • 所有对该对象的引用全部是强引用(除非我们显式地使用了:软引用(Soft Reference)、弱引用(Week Reference)或虚引用(Phantom Reference))。

    可能一些读者对这几种引用的概念还不是很清楚,下面分别对之加以介绍。

    对象引用的结构层次示意如图所示。

对象引用的结构层次示意

    强引用

    强引用(Strong Reference)是指JVM内存管理器从根引用集合(Root Set)出发遍寻堆中所有到达对象的路径。当到达某对象的任意路径都不含有引用对象时,对这个对象的引用就被称为强引用。

 

    软引用

    软引用(Soft Reference)的主要特点是具有较强的引用功能。只有当内存不够的时候,才回收这类内存,因此在内存足够的时候,它们通常不被回收。另外,这些引用 对象还能保证在Java抛出OutOfMemory异常之前,被设置为null。它可以用于实现一些常用资源的缓存,实现Cache的功能,保证最大限度 的使用内存而不引起OutOfMemory。下面是软引用的实现代码:

... ...
import java.lang.ref.SoftReference;
...
A a = new A();
...
// 使用a
...
// 使用完了a,将它设置为soft引用类型,并且释放强引用;
SoftReference sr = new SoftReference(a);
a = null;
    ...
    // 下次使用时
    if (sr != null) {
        a = sr.get();
    } else {
        // GC由于低内存,以释放a,因此需要重新装载;
        a = new A();
        sr = new SoftReference(a);
    }
    ... ...
 

 

    软引用技术的引进使Java应用可以更好地管理内存,稳定系统,防止系统内存溢出,避免系统崩溃(crash)。因此在处理一些占用内存较大而且声明周期 较长,但使用并不频繁的对象时应尽量应用该技术。正像上面的代码一样,我们可以在对象被回收之后重新创建,提高应用对内存的使用效率,提高系统稳定性。但 事物总是带有两面性的,有利亦有弊,在某些时候对软引用的使用会降低应用的运行效率与性能,例如:应用软引用的对象的初始化过程较为耗时,或者对象的状态 在程序的运行过程中发生了变化,都会给重新创建对象与初始化对象带来不同程度的麻烦,有些时候我们要权衡利弊择时应用。

 

    弱引用

    弱引用(Weak Reference)对象与Soft引用对象的最大不同就在于:GC在进行回收时,需要通过算法检查是否回收Soft引用对象,而对于Weak引用对 象,GC总是进行回收。Weak引用对象更容易、更块被GC回收。虽然,GC在运行时一定回收Weak对象,但是复杂关系的Weak对象群常常需要好几次 GC的运行才能完成。Weak引用对象常常用于Map结构中,引用占用内存空间较大的对象,一旦该对象的强引用为null时,对这个对象引用就不存在 了,GC能够快速地回收该对象空间。与软引用类似我们也可以给出相应的应用代码:

... ...
import java.lang.ref.WeakReference;
...
A a = new A();
...
// 使用a
...
// 使用完了a,将它设置为weak引用类型,并且释放强引用;
WeakReference wr = new WeakReference(a);
a = null;
    ...
    // 下次使用时
    if (wr != null) {
        a = wr.get();
    } else {
        a = new A();
        wr = new WeakReference(a);
    }
    ... ...
 

 

 

    虚引用

    虚引用(Phantom Reference)的用途较少,主要用于辅助finalize函数的使用。Phantom对象指一些执行完了finalize函数,并为不可达对象,但是还没有被GC回收的对象。这种对象可以辅助finalize进行一些后期的回收工作,我们通过覆盖Reference的clear()方法,增强资源回收机制的灵活性。

  注意     在实际程序设计中一般很少使用弱引用与虚引用,使用软引用的情况较多,这是因为软引用可以加速JVM对垃圾内存的回收速度,可以维护系统的运行安全,防止内存溢出(OutOfMemory)等问题的产生。

 

2.3  不可视阶段

    当一个对象处于不可视阶段,说明我们在其他区域的代码中已经不可以再引用它,其强引用已经消失,例如,本地变量超出了其可视范围,如下所示。

... ...
public void process() {
    try {
        Object obj = new Object();
        obj.doSomething();
    } catch (Exception e) {
        e.printStackTrace();
    }
    while (isLoop) { // ... loops forever
        // 这个区域对于obj对象来说已经是不可视的了
        // 因此下面的代码在编译时会引发错误
        obj.doSomething();
    }
}
... ...
 

 

    如果一个对象在使用完之后,而且在其可视区域不再使用,此时应该主动将其设置为空。可以在上面的代码行obj.doSomething();下添加代码行obj = null;这样一行代码强制将obj对象置为空值。这样做的意义是,可以帮助JVM及时地发现这个垃圾对象,并且可以及时地回收该对象所占用的系统资源。

 

2.4  不可到达阶段

    处于不可到达阶段的对象在虚拟机的对象引用根集合中再也找不到直接或间接的强引用,这些对象一般是所有线程栈中的临时变量,所有已装载的类的静态变量或者是对本地代码接口(JNI)的引用。

 

2.5  可收集阶段、终结阶段与释放阶段

    当一个对象处于可收集阶段、终结阶段与释放阶段时,该对象可能处于下面三种情况:

     (1) 回收器发现该对象已经不可到达。

     (2) finalize 方法已经被执行。

     (3) 对象空间已被重用。

    当对象处于上面三种情况下,该对象就处于可收集阶段、终结阶段与释放阶段了。

0
5
分享到:
评论

相关推荐

    实战JAVA虚拟机 JVM故障诊断与性能优化

    理解这些区域的作用以及对象的生命周期,对于理解和优化内存使用至关重要。 5. **垃圾收集机制**:GC的主要任务是回收不再使用的对象所占用的内存。包括 Minor GC(年轻代GC)、Major GC(老年代GC)和Full GC。...

    Java界面版 内存地址转换的三种方式过程演示

    例如,栈上分配策略将短生命周期的对象直接在虚拟机栈帧中分配,避免了堆内存的分配和回收。对象内存布局则涉及到实例字段、对齐填充等,JVM会根据这些信息计算出对象的实际内存地址。 在“Java界面版 内存地址转换...

    Java虚拟机-jvm故障诊断与性能优化-源码

    - **堆溢出**:过多的对象无法分配到内存中,可通过调整堆大小或优化对象生命周期解决。 - **栈溢出**:线程栈过深,可能由于递归调用或线程本地存储过多导致。 - **方法区溢出**:类加载过多,可以考虑限制加载...

    jvmjava,java实现的JVM。.zip

    这有助于深入理解JVM的生命周期管理、异常处理、多线程等复杂概念。 五、优化与进阶 掌握JVM的工作原理后,开发者可以进行更高级的调优,如调整JVM参数以优化内存分配、提高垃圾收集效率等。此外,还可以探索JVM的...

    JNI编程(二) —— 让C++和Java相互调用(2)

    在实际项目中,开发者需要注意一些关键点,比如内存管理(Java对象的生命周期由JVM管理,本地对象则需要手动管理),错误处理(Java和C++的异常处理机制不同),以及线程安全(多线程环境下,JNI方法的调用需要特别...

    java基础之JVM

    ### Java基础之JVM ...以上概述了JVM的基本概念及其核心组成部分,了解这些内容对于深入学习Java编程语言和提升编程效率至关重要。此外,掌握JVM的工作原理还有助于开发者更好地理解和优化Java应用程序的性能问题。

    深入java虚拟机(三)——类的生命周期(下)类的初始化1

    【深入Java虚拟机(三)——类的生命周期(下)类的初始化1】 类的生命周期在Java中是一个关键的概念,它涵盖了从加载到卸载的整个过程。在类的生命周期中,初始化阶段是非常重要的,因为它涉及到类的静态变量的赋值...

    java虚拟机源码-JVMbookSource:实战Java虚拟机———JVM故障诊断与性能优化(第2版)源码.rar

    《实战Java虚拟机——JVM故障诊断与性能优化(第2版)》是Java开发者深入理解JVM工作原理、诊断问题以及进行性能调优的重要参考资料。该书籍的源码提供了丰富的示例和实践案例,帮助读者更好地掌握Java虚拟机的内部...

    面试-Java一些常见面试题+题解之JVM-JVM.zip

    分代收集是现代JVM的主流策略,将堆分为新生代和老年代,根据对象生命周期的不同进行不同策略的垃圾回收。 4. **内存溢出与内存泄漏**:内存溢出是JVM无法分配新的内存,内存泄漏则指程序未释放不再使用的内存。...

    Java程序性能优化 让你的Java程序更快、更稳定

    2. **对象创建与内存管理**:减少不必要的对象创建,尤其是短生命周期的对象,可以降低垃圾收集的压力。合理使用对象池,避免频繁创建和销毁对象,有助于提高性能。 3. **数据结构与算法**:选择合适的数据结构和...

    精选_毕业设计_基于JAVA的内存管理模拟_完整源码

    在Java编程语言中,内存管理是一项至关重要的任务,它直接影响到程序的性能、稳定性和资源利用率。本毕业设计项目——“基于JAVA的内存管理模拟”深入探讨了Java内存模型及其管理机制,通过模拟实现来帮助理解这些...

    浅谈Java线程的生命周期——北大青鸟佳音旗舰.docx

    在Java编程中,线程是程序执行的最小单元,它们允许并发处理多个任务,提高程序的效率。本文将深入探讨Java线程的生命周期,包括创建、启动、结束以及线程的协作和调度。 首先,创建Java线程有两种主要方式:直接...

    java内存溢出

    Java作为一种广泛使用的编程语言,其自动内存管理机制——垃圾收集(Garbage Collection, GC)一直是其吸引开发者的重要特性之一。然而,这并不意味着Java程序就不会遇到内存相关的问题。事实上,Java同样存在内存...

    Java中堆内存和栈内存详解.doc

    在Java编程语言中,内存管理是一项核心技能。为了更好地理解和使用Java,了解其内存分配机制至关重要。本文将详细介绍Java中的两种主要内存区域——堆内存(Heap Memory)与栈内存(Stack Memory),并探讨它们之间...

    JVM内存资料.zip_jdk

    在Java中,对象主要在堆内存中分配,根据对象的生命周期和大小,分为新生代和老年代两个区域。新生代采用复制算法进行垃圾收集,老年代则使用标记-整理或分代收集算法。 对于大对象,JVM会直接在老年代分配,避免...

    EJB开发——Java 2 SDK.rar_java ppt

    1. EJB开发——Java 2 SDK.ppt:这是主教程文件,很可能包含了EJB在Java 2 SDK中的开发流程、生命周期管理、容器服务、会话Bean、实体Bean、消息驱动Bean、事务管理、安全性等方面的内容。用户可以期待学习如何在...

    Java并发编程实践.rar

    同时,讲解了Java中实现并发的基石——JVM内存模型,以及Java提供的并发工具包的基础知识。 第二章:Java并发基础 深入讨论Java中的线程创建与管理,包括Thread类的使用、Runnable接口的实现,以及线程的生命周期...

    Java当中的内存分配.pdf

    在Java虚拟机(JVM)的运行时数据区中,堆是用于存放对象实例的内存区域,而栈则负责存放局部变量、方法调用等。 首先,堆内存是Java内存管理的核心,所有通过new关键字创建的对象实例都存放在堆中。对象的创建实际...

    java面试——深圳-腾讯-Java高级.zip

    在准备深圳腾讯Java高级面试的过程中,你需要掌握一系列深入的Java编程和相关技术知识。这份压缩包文件"java面试——深圳-腾讯-Java高级.zip"包含了关键的面试指南,特别是对于那些寻求在大型科技公司如腾讯工作的...

    jvm分享ppt

    【JVM 分享】——深入理解Java程序的生命周期与执行机制 Java程序的生命周期始于源码,经过编译、加载和执行,最终在Java虚拟机(JVM)上运行。这个过程涉及到了编译器、类文件、虚拟机以及相关的优化技术。下面将...

Global site tag (gtag.js) - Google Analytics