java程序执行过程 +JVM内存管理 + GC垃圾回收机制

转载 2018年04月15日 19:54:29

java程序执行过程 +JVM内存管理 + GC垃圾回收机制

1、Java程序执行过程

一个java程序的编译和执行过程如下:

  • .java ——编译——> .class
  • 类加载器负责加载各个字节码文件(.class)
  • 加载完.class后,由执行引擎执行,在执行过程中,需要运行时数据区提供数据

这里写图片描述


补充:手动编译.java

Main.java

public class Main {
    public static void main(String[] args) {
        System.out.println("asdf");
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5

使用命令javac Main.java编译,编译后能看到一个.class文件,里面长这样

cafe babe:前四个字节称为魔数,作用是标识不同版本的虚拟机 
0000 0034:是java的版本号,0000是次版本号,0034是主版本号

  • 0x0034 = 3 * 16 + 4 = 52 表示JDK1。8版本

0013:是常量池入口,0x0013 = 19,表示常量池有18项常量(1~18,第0项空出来)

cafe babe 0000 0034 001d 0a00 0600 0f09
0010 0011 0800 120a 0013 0014 0700 1507
0016 0100 063c 696e 6974 3e01 0003 2829
5601 0004 436f 6465 0100 0f4c 696e 654e
756d 6265 7254 6162 6c65 0100 046d 6169
6e01 0016 285b 4c6a 6176 612f 6c61 6e67
2f53 7472 696e 673b 2956 0100 0a53 6f75
7263 6546 696c 6501 0009 4d61 696e 2e6a
...
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

最后使用命令java Main运行刚才的程序

关于.class文件的分析,请见:http://www.cnblogs.com/winner-0715/p/4935256.html



2、JVM内存管理

JVM将内存划分为6个部分:PC寄存器(也叫程序计数器)、虚拟机栈、堆、方法区、运行时常量池、本地方法栈

这里写图片描述


  • PC寄存器(程序计数器):用于记录当前线程运行时的位置,每一个线程都有一个独立的程序计数器,线程的阻塞、恢复、挂起等一系列操作都需要程序计数器的参与,因此必须是线程私有的。

  • java 虚拟机栈:在创建线程时创建的,用来存储栈帧,因此也是线程私有的。java程序中的方法在执行时,会创建一个栈帧,用于存储方法运行时的临时数据和中间结果,包括局部变量表、操作数栈、动态链接、方法出口等信息。这些栈帧就存储在栈中。如果栈深度大于虚拟机允许的最大深度,则抛出StackOverflowError异常。

    • 局部变量表:方法的局部变量列表,在编译时就写入了class文件
    • 操作数栈:int x = 1; 就需要将 1 压入操作数栈,再将 1 赋值给变量x
  • java 堆:java堆被所有线程共享,堆的主要作用就是存储对象。如果堆空间不够,但扩展时又不能申请到足够的内存时,则抛出OutOfMemoryError异常。

    StackOverflowErrorOutOfMemoryError
    java栈java堆
    栈深度超过范围了(比如:递归层数太多了)内存空间不够了(需要及时释放内存)


  • 方法区:方发区被各个线程共享,用于存储静态变量、运行时常量池等信息。

  • 本地方法栈:本地方法栈的主要作用就是支持native方法,比如在java中调用C/C++


3、GC回收机制

  • 哪些内存需要回收?——who
  • 什么时候回收?——when
  • 怎么回收?——how

1、哪些内存需要回收?

  • java堆、方法区的内存
线程私有线程共享
程序计数器、虚拟机栈、本地方法栈java堆、方法区
随线程生而生,随线程去而去。线程分配多少内存都是有数的,当线程销毁时,内存就被释放了堆和方法区的内存都是动态分配的(使用new关键字),所以也需要动态回收。
这部分内存的回收依赖GC完成


2、什么时候回收?

  • 引用计数法
  • 可达性分析

(1)、引用计数法

给对象添加一个引用计数器,每当有一个地方引用它时,计数器加一。反之每当一个引用失效时,计数器减一。当计数器为0时,则表示对象不被引用。

举个例子:

Object a = new Object(); // a的引用计数为1
a = null; // a的引用计数为0,等待GC回收
  • 1
  • 2

但是,引用计数法不能解决对象之间的循环引用,见下例

Object a = new Object(); // a的引用计数为1
Object b = new Object(); // b的引用计数为1

a.next = b; // a的引用计数为2
b.next = a; // b的引用计数为2

a = null; // a的引用计数为1,尽管已经显示地将a赋值为null,但是由于引用计数为1,GC无法回收a
b = null; // b的引用计数为1,同理,GC也不回收b
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8

(2)、可达性分析

设立若干根对象(GC Root),每个对象都是一个子节点,当一个对象找不到根时,就认为该对象不可达。

这里写图片描述

没有一条从根到Object4 和 Object5的路径,说明这两个对象到根是不可达的,可以被回收

补充:java中,可以作为GC Roots的对象包括:

  • java虚拟机栈中引用的对象
  • 方法区中静态变量引用的对象
  • 方法区中常量引用的对象
  • 本地方法栈中引用的对象


3、怎么回收?

  • 标记——清除算法
  • 复制算法
  • 分代算法

(1)、标记——清除算法

遍历所有的GC Root,分别标记处可达的对象和不可达的对象,然后将不可达的对象回收。

缺点是:效率低、回收得到的空间不连续

(2)、复制算法

将内存分为两块,每次只使用一块。当这一块内存满了,就将还存活的对象复制到另一块上,并且严格按照内存地址排列,然后把已使用的那块内存统一回收。

优点是:能够得到连续的内存空间 
缺点是:浪费了一半内存

这里写图片描述

(3)、分代算法

在java中,把内存中的对象按生命长短分为:

  • 新生代:活不了多久就go die 了,比如局部变量
  • 老年代:老不死的,活的久但也会go die,比如一些生命周期长的对象
  • 永久代:千年王八万年龟,不死,比如加载的class信息

有一点需要注意:新生代和老年代存储在java虚拟机堆上 ;永久代存储在方法区上

 回收方法
新生代使用复制算法
老年代使用标记——清除算法
永久代————————



补充:java finalize()方法:

在被GC回收前,可以做一些操作,比如释放资源。有点像析构函数,但是一个对象只能调用一次finalize()方法。

简单分析TXMLDocument内部结构

 第一次对TXMLDocument感兴趣,也是第一次使用TXMLDocument的时候,当时,我在修改一个软件,只是想加一点功能,由于时间不多,我决定先做一个接口,再慢慢扩充,但如何使接口扩充性能好,...
  • testnet
  • testnet
  • 2003-02-14 08:58:00
  • 733

面试总结:java程序执行过程 + JVM内存管理 + GC垃圾回收机制

java程序执行过程 +JVM内存管理 + GC垃圾回收机制1、Java程序执行过程一个java程序的编译和执行过程如下: .java ——编译——> .class 类加载器负责加载各个字节码文件(....
  • u010429424
  • u010429424
  • 2017-08-17 19:15:26
  • 1708

JVM——内存管理和垃圾回收

GC   何为GC? Java与C语言相比的一个优势是,可以通过自己的JVM自动分配和回收内存空间。垃圾回收机制是由垃圾收集器Garbage Collection GC来实现的,GC是后台的守护进程...
  • SEU_Calvin
  • SEU_Calvin
  • 2016-07-12 21:05:30
  • 15885

JVM内存管理及GC机制

一、概述Java GC(Garbage Collection,垃圾收集,垃圾回收)机制,是Java与C++/C的主要区别之一,作为Java开发者,一般不需要专门编写内存回收和垃圾清理代码,对内存泄露和...
  • suifeng3051
  • suifeng3051
  • 2015-09-08 16:28:18
  • 14123

JVM架构和GC垃圾回收机制(JVM面试不用愁)

JVM架构和GC垃圾回收机制 JVM架构图分析 JVM被分为三个主要的子系统: 1.  类加载器子系统 2.  运行时数据区 3.  执行引擎 1. 类加载器子系统 Java的动态类加载功能是由类加...
  • aijiudu
  • aijiudu
  • 2017-06-10 16:49:15
  • 3473

JVM内存管理机制和垃圾回收机制

从Java平台的逻辑结构上来看,我们可以从下图来了解JVM:从上图能清晰看到Java平台包含的各个逻辑模块,也能了解到JDK与JRE的区别对于JVM自身的物理结构,我们可以从下图鸟瞰一下:对于JVM的...
  • u011225629
  • u011225629
  • 2015-10-09 15:06:04
  • 5603

javascript的垃圾回收机制与内存管理

一、垃圾回收机制—GC Javascript具有自动垃圾回收机制(GC:Garbage Collecation),也就是说,执行环境会负责管理代码执行过程中使用的内存。 原理:垃圾收集器会...
  • OLiver_web
  • OLiver_web
  • 2016-12-31 17:05:21
  • 4438

深入理解JVM(六)-Java垃圾回收机制GC

对于C/C++开发者来说,他们在内存管理方面具有至高的权利,但是也承担着巨大的维护责任。而对于Java程序员来说,有了JVM(Java虚拟机)管理机制的帮助,再也不用担心内存泄漏和内存溢出问题了。因此...
  • dingji_ping
  • dingji_ping
  • 2016-03-31 20:51:22
  • 1423

JVM GC(垃圾回收机制)

写文章 Java性能优化之JVM GC(垃圾回收机制) Java的性能优化,整理出一篇文章,供以后温故知新。 JVM GC(垃圾回收机制) 在学习Java GC 之前,...
  • sinat_34979383
  • sinat_34979383
  • 2017-03-27 08:03:40
  • 779

JVM 垃圾回收机制主要原理

对于垃圾JVM的垃圾回收机制这里我们称为GC,众所周知,java语言不需要像c++那样需要自己申请内存,自己释放内存,这些都是JVM帮我们做好了的,但是对于一名java程序员,想要更近自己的水平更上一...
  • qq_33048603
  • qq_33048603
  • 2016-10-03 14:21:09
  • 14532
收藏助手
不良信息举报
您举报文章:java程序执行过程 +JVM内存管理 + GC垃圾回收机制
举报原因:
原因补充:

(最多只允许输入30个字)