jvm系列(1)java虚拟机的基本结构

 

1.类加载子系统:从文件系统或者网络中加载class信息,而加载的这些信息就会存放一块称之为在方法区的内存空间中。

2.方法区:就是存放类信息,各种常量信息(  对于Java8, HotSpots取消了永久代,那么是不是也就没有方法区了呢?当然不是,方法区是一个规范,规范没变,它就一直在。那么取代永久代的就是元空间。它可永久代有什么不同的?存储位置不同,永久代物理是是堆的一部分,和新生代,老年代地址是连续的,而元空间属于本地内存;存储内容不同,元空间存储类的元信息,静态变量和常量池等并入堆中。相当于永久代的数据被分到了堆和元空间中。)。

3.java堆:它是java程序最主要的内存工作区域,几乎所有的对象实例都存放在java堆中,堆空间是所有线程共享的。在java虚拟机启动的时候就会建立java堆。(堆内存用来存放new创建的对象和数组。堆内存中所有的实体都有内存地址值。堆内存中的实体是用来封装数据的,这些数据都有默认初始化值。堆内存中的实体不再被指向时,JVM启动垃圾回收机制,自动清除,这也是JAVA优于C++的表现之一(C++中需要程序员手动清除))

4.直接内存:java的NIO库允许java程序使用直接内存,从而达到提高性能。直接内存的读写速度比java堆要好,所以可以考虑在读写频繁的场合使用直接内存。

5.java栈:每个虚拟机线程都有一个私有的栈,一个线程的java栈在线程被创建的是被创建,java栈中有局部变量,方法参数,返回值,方法调用等。(函数中定义的基本类型变量,对象的引用变量都在函数的栈内存中分配。栈内存特点,数据一执行完毕,变量会立即释放,节约内存空间。栈内存中的数据,没有默认初始化值,需要手动设置。)

6.本地方法栈:和java栈差不多,但是本地方法栈用于本地方法调用,java虚拟机允许java直接调用本地方法(通常是c编写)。

7.垃圾回收系统:是java的核心,java自己有一套自己的垃圾回收机制,我们无需手动去清理,垃圾回收器可以对方法区、Java堆和直接内存进行回收。

8.pc寄存器:是每个线程的私有空间。每个线程都会被java虚拟机创建pc寄存器,在任何时刻,一个java线程总是在执行一个方法,这个叫当前方法,如果当前方法不是本地方法,pc寄存器就会执行当前正在被执行的指令,如果是本地方法,则pc寄存器值为undefined,寄存器存放如当前环境指针,程序计数器,操作栈指针等信息。

9.执行引擎:它负责执行虚拟机的字节码。一般会先编译成机器码后执行。

 

2. 堆空间的一般结构
2.1 结构图

 绝大多数情况下,对象首先分配地eden区,在一次新生代回收后,如果对象还存活,则会进入s0或者s1之后,每经过一次新生代回收,对象如果还存在,它的年龄就会加1 。当对象的年龄达到一定条件后,就会认为是老年对象,从而进入老年代。

2.2 例子

堆、方法区、栈的关系


3. 出入Java栈
1. Java栈是一块线程私有的内存空间。如果说,Java堆和程序数据密切相关,那么栈就是和线程执行密切相关。线程执行的基本行为是函数调用,每次函数调用的数据是通过Java栈传递的。

2. Java栈中保存的主要内容为栈帧。每一次函数调用,都会有一个对应的栈帧被压入Java栈,每一个函数调用结束,都会有一个栈帧被弹出Java栈。

3. JaVa栈方法有两种返回函数的方式,一种是正常的函数返回,使用return指;另外一种是抛出异常。

3.1 栈帧和函数调用

1.        由于每次函数调用都会生成对应的栈帧,从而占用一定的栈空间,因此,如果栈空间不足,那么函数调用自然无法继续进入下去。当请求的栈深度大于最大可用栈深度时,系统就会抛出StackOverflowError            栈溢出错误。

2.        函数嵌套调用的层次在很大程度上由栈的大小,栈越大,函数可以支持的嵌套调用次数就越多。

3.2局部变量表
1.        局部变量用于保存函数的参数以及局部变量。局部变量表中的变量只在当前函数调用中有效,当函数调用结束后,随着函数栈帧的销毁,局部变量表也会随之销毁。

2.        局部变量表在栈帧之中,因此,如果函数的参数和局部变量较多,会使得局部变量表膨胀,从而每一次函数调用就会占用更多的栈空间,最终导致函数的嵌套调用次数减少。

3.        栈帧中的局部变量表中的槽位是可以重复的,如果一个局部变量过了其作用域,那么在其作用域之后申明的新的局部变量就很有可能会复用过期局部变量的槽位,从而达节省资源的目的。

4.        局部变量表中的变量也是重要的垃圾回收根节点,只要被局部变量表中直接或间接引用的对象都是会被回收的。

3.3 操作数栈
操作数数也是一个先进后出的数据结构,只支持入栈和出栈两种操作。它主要用于保存计算的中间结果,同时作为计算过程中变量临时的存储空间。


3.4帧数据区
除了局部变量表和操作数外,Java栈帧还需要一些数据来支持常量池解析、正常方法返回和异常处理等。大部分Java字节指令需要进行常量池访问,在帧数据区中保存着访问常量池的指针,方便程序访问常量池。

3.5 栈上分配
1.        栈上分配是Java虚拟机提供的一项优化技术,它的基本思想是,对于那些线程私有的对象(这里不可能被其他线程访问的对象),可以将它们打散分配在栈上,而不是分配在堆上。分配在栈上的好处是可以在函数调用结束后自行销毁,而不是需要垃圾回收器的介入,从而提高系统的性能。

2.        栈上分配的一个技术基础是进行逃逸分析。逃逸分析的目的是判断对象的作用域是否可能逃逸出函数体。

以下方法属性逃逸对象

<span style="font-weight: normal;">private static User u;
    public static void alloc()
    {
        u = new User();
        u.id = "5";
        u.name = "gegym";
    }</span>
以下方法属性非逃逸对象


public static void alloc()
    {
        User u = new User();
        u.id = "5";
        u.name = "gegym";
    }

 


--------------------- 
作者:Owen William 
来源:CSDN 
原文:https://blog.csdn.net/owen_william/article/details/50987640 
版权声明:本文为博主原创文章,转载请附上博文链接!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java虚拟机结构主要由类加载器、虚拟机栈、本地方法栈、堆、方法区组成。类加载器的作用是加载Java类;虚拟机栈是用于存储局部变量、操作数栈等信息;本地方法栈是用于存储本地方法的栈;堆是Java虚拟机使用的最大内存空间,用于存储对象实例;而方法区则存放类信息、常量、静态变量等数据。 ### 回答2: Java虚拟机Java Virtual Machine,JVM)是Java程序的运行环境,它在物理机器上创建一个虚拟的计算机平台,在这个平台上执行Java字节码。JVM结构包含以下几个关键组成部分: 1. 类加载器(ClassLoader):负责加载字节码文件(.class文件)并将其转换为Java类的内存表示。类加载器可以根据需要动态加载和卸载类。 2. 执行引擎(Execution Engine):负责执行Java字节码。执行引擎将字节码解释为机器指令序列或将其编译为本地代码执行,以达到提高性能的目的。 3. 运行时数据区(Runtime Data Areas):包括多个不同类型的数据区域,用于存储程序运行所需的数据。 - 方法区(Method Area):存储被加载的类信息、常量、静态变量、即时编译器编译后的代码等。 - 堆(Heap):存储Java对象实例,堆是在JVM启动时创建的,用于存放动态分配的对象。 - 栈(Stack):存储方法执行时的局部变量、操作数栈、调用信息等。每个线程都有自己的栈,用于方法的调用和返回。 - 程序计数器(Program Counter Register):记录当前线程执行的指令地址或指令索引。 4. 本地方法接口(Native Method Interface,JNI):允许Java应用程序调用使用其他语言编写的本地库中的方法。 5. 安全性引擎(Security Engine):提供安全管理和访问控制功能,确保Java程序在执行时具有必要的权限。 这些组成部分共同构成了Java虚拟机结构Java程序在JVM上运行时,通过类加载器将程序转化为内存表示,在运行时数据区执行代码,执行引擎解释和执行字节码指令,最终完成Java程序的运行。JVM结构和功能的设计有效地将Java程序的开发与底层的操作系统解耦,提供了跨平台的能力。 ### 回答3: Java虚拟机JVM)是Java程序的运行环境,它是一个软件程序,能够解释和执行Java字节码。JVM结构可以分为以下几个部分: 1. 类加载器(Class Loader):类加载器负责加载Java类文件,将其加载到内存中,并生成对应的类对象。JVM中有三个主要的类加载器:启动类加载器(Bootstrap Class Loader)、扩展类加载器(Extension Class Loader)和应用程序类加载器(Application Class Loader)。 2. 运行时数据区(Runtime Data Area):运行时数据区是JVM用于存储程序运行时所需数据的区域。主要包括方法区、堆、虚拟机栈、本地方法栈和程序计数器。 - 方法区(Method Area):用于存储类的结构信息、静态变量、常量等数据。 - 堆(Heap):用于存储Java对象。所有的对象实例都分配在堆中,并可以通过引用在方法区或栈中访问。 - 虚拟机栈(VM Stack):每个线程在运行时都会创建一个对应的虚拟机栈,用于存储局部变量、方法参数、返回值等。 - 本地方法栈(Native Method Stack):与虚拟机栈类似,但用于执行本地方法。 - 程序计数器(Program Counter):用于记录当前线程执行的字节码指令地址。 3. 执行引擎(Execution Engine):执行引擎负责解释和执行字节码指令,将其转换为对应的机器指令。常见的执行引擎有两种:解释器(Interpreter)和即时编译器(Just-In-Time Compiler,JIT)。 4. 本地方法接口(Native Interface):本地方法接口提供了Java代码调用本地方法的能力。本地方法接口定义了一组规范,使得Java代码可以与C、C++等底层语言进行交互。 5. 垃圾回收系统(Garbage Collection System):垃圾回收系统负责自动管理堆内存的分配和释放,回收不再使用的对象。垃圾回收系统通过标记-清除、复制算法等方式来回收内存。 通过以上的结构Java虚拟机能够提供一种平台无关的执行环境,使得Java程序在不同的操作系统和硬件平台上都能够运行。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值