关于程序员:浅谈JVM内存结构

内存是操作系统中不可或缺的局部,一台机器有内存能力失常稳固地运行。编程语言从来都有本人的内存管理机制,尤其是Java,它不像C一样须要本人保护内存的关系,而是通过本人的外部机制JVM来治理内存。这尽管升高了开发同学们的动手难度,但同时也使得在运行时一旦抛出内存异样,很难晓得产生了什么。以下就简略地来介绍一下JVM内存的构造。

废话不多说,先看上面图表:

▲图表1 JVM整体构造▲

此时可能有些人会一头雾水,别着急,咱们一点点来看。

先看第一局部:

▲图表2 类装载零碎▲

这个代表了一个类被装入JVM的过程。如果用更简略的比喻,它相似初学JDBC时Class.forName()所做的事件。具体来讲,会分为以下几步:

  1. 加载:将字节码文件依照双亲委托机制(当某个类加载器须要加载某个.class文件时,它首先把这个工作委托给他的下级类加载器,递归这个操作。如果下级的类加载器没有加载,本人才会去加载这个类,方块中对应的便是Bootstrap Class Loader(启动类加载器),Extension Class Loader (规范扩大类加载器),Application Class Loader(零碎类加载器))进行加载;
  2. 链接字节码文件:分为三个步骤,别离是字节码验证(verify)、class类数据结构剖析(prepare)以及相应的内存调配和最初的符号表的链接(resolve);
  3. 初始化操作:比方类中动态属性和初始化赋值,以及动态块的执行等。

一个类就是这样被装入了内存,那么在程序中会产生什么呢?别着急,且听我缓缓道来:

▲图表3 Java内存构造▲

这张图是运行时数据区,示意了以后JVM的所有状态,让咱们一个区一个区看看:

1. 办法区(Method Area):用于存储已被虚拟机加载的类信息、常量、动态变量、即时编译器编译后的代码。运行时常量池也是办法区的一部分,比方String w = ”hello”;中,hello就被放在了办法区里。办法区是线程共享的。有一点要留神,JDK1.8 应用元空间 MetaSpace 代替办法区,元空间并不在 JVM中,而是应用本地内存;

_2. 堆区(Heap Area):_堆区是JVM中占地最大的区域,所有的实例对象全副都在堆区上,这个地位也是线程共享的;

3. 栈区(Stack Area):寄存了每一个线程的以后状态,每一个线程都有一个本人的栈,而栈中寄存了以下数据组成的一个个栈帧:操作数、局部变量表、动静链接、返回地址,须要留神的是,栈中只存援用或者根本类型,而且线程不共享(并没有指外部的优化动作);

4. 程序计数器(PC Registers): 它是以后线程执行字节码的行号指示器。在多线程中,为了让每个线程切换回来后可能复原原来执行的指令,就须要为每个线程启动一个PC计数器,这些计数器之间是互补影响的,因为程序计数器和栈一样都是线程公有的。当然程序计数器是JVM惟一个不会呈现内存溢出的组件;

5. 本地办法栈(Native Method Statck):保留了本地办法,它是当程序调用类库(本地办法)中的办法时才会用到它,即native method。

接下来就要介绍为咱们勤勤恳恳工作的执行引擎啦:

▲图表4 Java执行引擎▲

Java执行分为编译执行(JIT compilation)解释执行(Interpreter)

首先咱们要明确什么是编译执行,什么是解释执行:

Bash就是属于解释执行语言,一行行解释代码来实现命令;

C就是编译执行, 将文件编译成字节码,接着运行。

那为什么Java会用两套编译伎俩呢?先看上面这个例子:

假设你是导演,写了个剧本,让演员表演。

一种形式是让演员把整个剧本都背下来,吃透到脑子里,而后间断表演一个小时。

另一种形式是让演员表演两分钟,再看两分钟脚本,思考一下,再表演两分钟,再看一会脚本,思考一下…

单纯从效率上来讲,第一种形式肯定会比第二种形式表演起来更纯熟,然而事实往往不容许,或者不必要。如果不是十分须要表演的技巧,简略地看一下剧本就好啦。在特地考验表演技巧时,才须要背下整个剧本,这样能力在表演时更好地展示本人的风采。

Java也是如此,由JIT发现热代码后,将指令集优化(比方重排,合并),而后生成字节码供零碎运行。至于其余的代码呢,简略的解释执行完就好了~

在运行的过程中肯定会产生很多的垃圾,因为随着零碎运行,很多对象都会废除不必,此时就须要应用垃圾回收机制Garbage Collection(垃圾回收内容太多了,当前能够单拿进去讲讲)。

最初,来看下Java与零碎底层交互:

▲图表5 Java与零碎底层交互▲

JNI(Java本地接口)通过应用Java本地接口书写程序,能够确保代码在不同的平台上不便移植。通过JNI实现与本地办法库的调用交互,使得在Java虚拟机内运行的Java代码可能与其它编程语言相互操作,包含_创立本地办法、更新Java对象、调用Java办法,_援用Java类,捕获和抛出异样等,也容许Java代码调用 C/C++或汇编语言编写的库。

好啦,明天就到这里,置信大家对JVM的内存模型曾经有了肯定的认知,心愿在遇到这种问题时,本篇内容能够帮到正在浏览的你。学习技术是一个与日俱增的过程,切不可浮躁~如果写代码有天意,那肯定是让你修炼!

【腾讯云】轻量 2核2G4M,首年65元

阿里云限时活动-云数据库 RDS MySQL  1核2G配置 1.88/月 速抢

本文由乐趣区整理发布,转载请注明出处,谢谢。

您可能还喜欢...

发表回复

您的电子邮箱地址不会被公开。 必填项已用*标注

此站点使用Akismet来减少垃圾评论。了解我们如何处理您的评论数据