`
yangjb
  • 浏览: 32071 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

jvm(转)

    博客分类:
  • java
阅读更多
在<<Java编程思想>>书中,作者对垃圾回收的精要概述:"基于系统平台自适应的停止-复制"垃圾回收技术和"sun早期自适应的标记-清扫"垃圾回收技术。当然,对一个系统来说,初始化和清理是系统整个运行过程中大两大核心"进程",我觉得"初始化"和"清理"已经称为一种专业术语,不管是小段代码,一个大程序,一个系统,甚至是软件架构,在其本身构建和运行过程中,垃圾回收如同模拟现实世界场景一样,已称为一种模式,一种"行为"或是对象的"习惯"。

    模拟现实场景来说,人是如何清理垃圾的,被丢弃的废纸,废品... 人会有专门的垃圾清理站或是保洁公司来处理这些“已耗尽”的能量,使其在土壤中循环再造。然而,这些垃圾也可以看做是"现实对象"。然而要模拟到用计算机实现这种垃圾回收系统,定是很有趣的事情,或许你早已在脑海中构建一副用计算机实现的垃圾回收场景,可那画面的内幕还是需要细细设计。于是有兴趣参考优化编程,java垃圾回收有向图等资料,简单地整理了一些基本的概念,分享之。若是很了解垃圾回收内幕的高手或是大师,还望你指点,我是这方面的新手,还在门前虚心求教和批评指正。

1. 垃圾回收

   垃圾回收是Java程序设计中内存管理的核心概念,JVM的内存管理机制被称为垃圾回收机制。

  一个对象创建后被放置在JVM的堆内存中,当永远不再引用这个对象时,它将被JVM在堆内存中回收。被创建的对象不能再生,同时也没有办法通过程序语句释放它们。即当对象在JVM运行空间中无法通过根集合到达(找到)时,这个对象被称为垃圾对象。根集合是由类中的静态引用域与本地引用域组成的。JVM通过根集合索引对象。

    在做Java应用开发时经常会用到由JVM管理的两种类型的内存:堆内存和栈内存。简单来讲,堆内存主要用来存储程序在运行时创建或实例化的对象与变量。例如通过new关键字创建的对象。而栈内存则是用来存储程序代码中声明为静态或非静态的方法。

(1) 堆内存

    堆内存在JVM启动的时候被创建,堆内存中所存储的对象可以被JVM自动回收,不能通过其他外部手段回收,也就是说开发人员无法通过添加相关代码的手段来回收堆内存中的对象。堆内存通常情况下被分为两个区域:新对象区域与老对象区域。

    新对象区域:又可细分为三个小区域:伊甸园区域、From区域与To区域。伊甸园区域用来保存新创建的对象,它就像一个堆栈,新的对象被创建,就像指向该栈的指针在增长一样,当伊甸园区域中的对象满了之后,JVM系统将要做到可达性测试,主要任务是检测有哪些对象由根集合出发是不可达的,这些对象就可以被JVM回收,并且将所有的活动对象从伊甸园区域拷贝到To区域,此时一些对象将发生状态交换,有的对象就从To区域被转移到From区域,此时From区域就有了对象。上面对象迁移的整个过程,都是由JVM控制完成的。

    老对象区域:在老对象区域中的对象仍然会有一个较长的生命周期,大多数的JVM系统垃圾对象,都是源于"短命"对象,经过一段时间后,被转入老对象区域的对象,就变成了垃圾对象。此时,它们都被打上相应的标记,JVM系统将会自动回收这些垃圾对象,建议不要频繁地强制系统作垃圾回收,这是因为JVM会利用有限的系统资源,优先完成垃圾回收工作,导致应用无法快速地响应来自用户端的请求,这样会影响系统的整体性能。

(2) 栈内存

    堆内存主要用来存储程序在运行时创建或实例化的对象与变量。例如通过new关键字创建的对象。而栈内存则是用来存储程序代码中声明为静态或非静态的方法。

2. JVM中对象的生命周期

   在JVM运行空间中,对象的整个生命周期大致可以分为7个阶段:
   创建阶段;
   应用阶段;
   不可视阶段;
   不可到达阶段;
   可收集阶段;
   终结阶段;
   释放阶段 

   上面这7个阶段,构成了JVM中对象的完整的生命周期。


   (1) 创建阶段

       在对象的创建阶段,系统主要通过下面的步骤,完成对象的创建过程:
    
       <1> 为对象分配存储空间;
       <2> 开始构造对象;
       <3> 从超类到子类对static成员进行初始化;
       <4> 超类成员变量按顺序初始化,递归调用超类的构造方法;
       <5> 子类成员变量按顺序初始化,子类构造方法调用。

       在创建对象时应注意几个关键应用规则:
      
       <1> 避免在循环体中创建对象,即使该对象占用内存空间不大。
       <2> 尽量及时使对象符合垃圾回收标准。比如 myObject = null。
       <3> 不要采用过深的继承层次。
       <4> 访问本地变量优于访问类中的变量。

      
Java代码 
1.比如:  
2.       for (int i = 0; i < 10000; i++) {  
3.             Object obj = new Object();  
4.             System.out.println(obj);  
5.       }  
6. 
7.       和  
8.       Object obj = null;  
9.       for (int i = 0; i < 10000; i++) {  
10.             obj = new Object();  
11.             System.out.println(obj);  
12.       } 
比如:
       for (int i = 0; i < 10000; i++) {
             Object obj = new Object();
             System.out.println(obj);
       }

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

       这里修正一下,在绝对情况下,效率还是上面的要好, 因为后面的多了一条初始化语句。
        而且各自作用域不同。 谢谢各位~~
   (2) 应用阶段

       在对象的引用阶段,对象具备如下特征:

      <1> 系统至少维护着对象的一个强引用(Strong Reference);
      <2> 所有对该对象的引用全部是强引用(除非我们显示地适用了:软引用(Soft Reference)、弱引用(Weak Reference)或虚引用(Phantom Reference)).

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

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

                              
Java代码 
1.下面是软引用的实现代码:  
2. 
3.                                import java.lang.ref.SoftReference;  
4.                                ...  
5.                                  
6.                                A a = new A();  
7.                                ...  
8. 
9.                                // 使用a  
10.                                ...  
11.                                   
12.                                // 使用完了a, 将它设置为soft引用类型,并且释放强引用  
13.                                SoftReference sr = new SoftReference(a);  
14.                                a = null;  
15.                                ...  
16. 
17.                                // 下次使用时  
18.                    if (sr != null) {  
19.                    a = sr.get();  
20.                } else {  
21.                    // GC由于低内存,已释放a,因此需要重新装载  
22.                                    a = new A();  
23.                    sr = new SoftReference(a);  
24.                } 
下面是软引用的实现代码:

                                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应用可以更好地管理内存,稳定系统,防止系统内存溢出,避免系统崩溃。因此在处理一些占用内存较大且生命周期较长,但使用并不繁地对象时应尽量应用该技术。提高系统稳定性。
            
                                     
       弱引用(Weak Reference):弱应用对象与软引用对象的最大不同就在于:GC在进行垃圾回收时,需要通过算法检查是否回收Soft应用对象,而对于Weak引用,GC总是进行回收。Weak引用对象更容易、更快地被GC回收。Weak引用对象常常用于Map结构中。
                             
Java代码 
1.下面是弱引用的实现代码:  
2. 
3.                               import java.lang.ref.WeakReference;  
4.                               ...  
5.                                 
6.                               A a = new A();  
7.                               ...  
8. 
9.                               // 使用a  
10.                               ...  
11.                                  
12.                               // 使用完了a, 将它设置为Weak引用类型,并且释放强引用  
13.                               WeakReference wr = new WeakReference(a);  
14.                               a = null;  
15.                               ...  
16. 
17.                               // 下次使用时  
18.                if (wr != null) {  
19.                    a = wr.get();  
20.            } else {  
21.                                   a = new A();  
22.                wr = new WeakReference(a);  
23.            } 
下面是弱引用的实现代码:

                                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 Reference)对象指一些执行完了finalize函数,并为不可达对象,但是还没有被GC回收的对象。这种对象可以辅助finalize进行一些后期的回收工作,我们通过覆盖了Refernce的clear()方法,增强资源回收机制的灵活性。

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


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

     
Java代码 
1.try {  
2.            Object localObj = new Object();  
3.     localObj.doSomething();  
4.      } catch (Exception e) {  
5.          e.printStackTrace();  
6.      }  
7. 
8.      if (true) {  
9.    // 此区域中localObj 对象已经不可视了, 编译器会报错。  
10.    localObj.doSomething();  
11.      } 
try {
             Object localObj = new Object();
     localObj.doSomething();
       } catch (Exception e) {
           e.printStackTrace();
       }

       if (true) {
    // 此区域中localObj 对象已经不可视了, 编译器会报错。
    localObj.doSomething();
       }   

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


   (5) 可收集阶段、终结阶段与释放阶段
       当一个对象处于可收集阶段、终结阶段与释放阶段时,该对象有如下三种情况:

       <1> 回收器发现该对象已经不可达。

       <2> finalize方法已经被执行。

       <3> 对象空间已被重用。

分享到:
评论

相关推荐

    neo4j-graphql-java:用于GraphQL查询和Neo4j的Cypher突变的纯JVM转换

    neo4j-graphql-java:用于GraphQL查询和Neo4j的Cypher突变的纯JVM转换

    JVM调优实战(转)

    网上下载的JVM调优实战, 值得一下

    JVM指令码.htm

    JVM指令码表,JVM运行原理学习的必备工具。常量入栈指令、局部变量值转载到栈中指令、将栈顶值保存到局部变量中指令、wide指令、通用(无类型)栈操作指令、类型转换指令、整数运算、浮点运算等指令。

    java中jvm原理和实现

    JVM(Java Virtual Machine)是实现Java程序运行的核心部分,它是一个虚拟机,负责将Java字节码转换为机器码并执行。涉及到类加载器、运行时数据区、执行引擎和垃圾收集器等组件。JVM通过加载字节码文件,将其转换为...

    深入理解JVM内存结构及运行原理全套视频加资料.txt

    2019最新深入理解JVM内存结构及运行原理(JVM调优)高级核心课程视频教程下载。JVM是Java知识体系中的重要部分,对JVM底层的了解是每一位Java程序员深入Java技术领域的重要因素。本课程试图通过简单易懂的方式,系统...

    最新java面试专题01-JVM

    最新jvm面试题合集,涵盖JVM运行时数据区、垃圾回收算法、垃圾回收器、类加载机制、JIT即时编译等核心知识点及常见面试题,一书在手,天下我有。 JVM内存结构:JVM的内存结构主要包括堆内存、方法区、栈(包括Java...

    jvm.go:用Go编写的玩具JVM

    转到1.12 编译jvm.go git clone https://github.com/zxh0/jvm.go.git cd jvm.go go build github.com/zxh0/jvm.go/cmd/java 使用Java安装运行jvm.go 检查您的Java版本,并确保已设置JAVA_HOME env ./java -...

    Java(JVM)虚拟机结构基础(转自Java研究组织)

    简略介绍jvm 不错

    JVM参数设置详细说明

    get、set方法转成本地代码 d: -XX:+PrintGCDetails 打应垃圾收集的情况如: [GC 15610.466: [ParNew: 229689K-&gt;20221K(235968K), 0.0194460 secs] 1159829K-&gt;953935K(2070976K), 0.0196420 secs] e: -XX:+...

    深入理解Java虚拟机视频教程(jvm性能调优+内存模型+虚拟机原理)视频教程

    第5节jvm再体验-jvm可视化监控工具 [免费观看] 00:21:17分钟 | 第6节杂谈 [免费观看] 00:12:37分钟 | 第7节Java的发展历史00:27:24分钟 | 第8节Java的发展历史续00:02:27分钟 | 第9节Java技术体系00:08:46分钟 |...

    jvm2cfg:原型Python脚本,可将JVM字节码转换为控制流程图

    jvm2cfg Python原型脚本将JVM字节码转换为控制流程图。 Utlize networkx框架来构建图形,并使用matplotlib显示可视化表示。 支持方法之间的函数调用边。 要运行,请在jvm2cfg.py的第267行上更改名为“ text_file”的...

    JVM常见面试题.docx

    JVM(Java虚拟机)是Java编程语言的运行时环境,它负责将Java字节码转换为机器码并执行程序。在Java开发岗位的面试中,JVM常见面试题主要涉及Java虚拟机的原理、内存管理、垃圾回收、类加载机制等方面。

    深入理解_Java_虚拟机 JVM_高级特性与最佳实践

    / 112 5.2.5 服务器JVM进程崩溃 / 113 5.3 实战:Eclipse运行速度调优 / 114 5.3.1 调优前的程序运行状态 / 114 5.3.2 升级JDK 1.6的性能变化及兼容问题 / 117 5.3.3 编译时间和类加载时间的优化 / 122 5.3.4 ...

    railgun:nailgun服务器的Ruby客户端(https

    大多数(通常是全部)成本来自每次调用的JVM旋转。 nailgun ( )是解决此问题的聪明方法。 这个想法是让JVM在加载所有必需类的情况下运行,并使用瘦客户机执行任意命令。 请参阅nailgun项目以获取更多信息。为什么...

    JVM虚拟机内容导图.xmind

    JVM虚拟机各知识点总结整理, 包括java虚拟机概念、堆、栈、方法区、垃圾回收概念、算法及分代转换、垃圾收集器参数配置、算法实现等各方面内容,每个点都有详细的备注描述介绍

    开发常见jvm问题调优.doc

    开发常见jvm问题调优.doc

    leetcode分类-leetcodePlan:刷题记录

    java是先将编码转换成字节码文件,再通过JVM转换为二进制码,再执行输出 b. c是直接将编码转换为二进制码,再由计算机去识别输出 3. java和c的内存管理区别 a. 在java有自动回收内存的机制,不用考虑内存的使用...

    CrossMobile:使用Java创建本机iOS,Android,Windows Phone和桌面应用程序。 编写一次,并生成复杂的多平台应用程序

    iOS :使用工具的最新版本将代码从JVM转换为ObjectiveC。 Android :具有适用于iOS API的精简兼容性库,并将所有系统调用传递给Android子系统,从而从开发人员中删除了Android实现细节。 桌面:采用类似于Android...

    Spring加载的:Java代理,可在运行中的JVM中重新加载类

    Spring Loaded是一个JVM代理,用于在JVM运行时重新加载类文件更改。 它在加载时转换类,以使其适合以后重新加载。 与“热代码替换”不同,后者仅允许在JVM运行后进行简单更改(例如,更改方法主体),而Spring ...

    组装:将WebAssembly编译为JVM和其他WASM工具

    Asmble是将代码编译为JVM字节码的编译器。 它还包含用于从命令行和JVM语言处理WASM代码的解释器和实用程序。 特征 WASM到JVM字节码编译器(无需运行时) WASM解释器(一次可逐步执行的指令) WASM二进制,WASM文...

Global site tag (gtag.js) - Google Analytics