java内存区域的划分
前言
之前我们探讨过一个.class文件是如何被加载到jvm中的。但是jvm内又是如何划分内存的呢?这个内被加载到了那一块内存中?jvm内存划分也是面试当中必被问到的一个面试题。
什么是JVM内存区域划分?
其实这个问题非常简单,JVM在运行我们写好的代码时,他是必须使用多块内存空间的,不同的内存空间用来放不同的数据,然后配合我们写的代码流程,才能让我们的系统运行起来。
举个最简单的例子,比如咱们现在知道了JVM会加载类到内存里来供后续运行,那么我问问大家,这些类加载到内存以后,放到哪儿去了呢?想过这个问题吗?
所以JVM里就必须有一块内存区域,用来存放我们写的那些类。
包括我们定义的成员变量,类变量,方法,局部变量等等,都在jvm内存中对应着一块内存来记录存储。
存放类的方法区(元数据空间)
在JDK1.8之前的版本里,代表JVM的一块区域。在1.8版本以后,这块区域的名字改了,叫做“Matespace”,可以认为是“元数据空间”这样的意思,当然这里主要存放的还是我们自己写的各种类的相关信息。
举个栗子。有如下两个类,People类没有成员变量,而Student类有一个name的类变量。
public class Student{
private static String name = "lisi";
}
public class People{
public static void main(){
Student student = new Student();
}
}
这两个类被加载到JVM,就会存放在这个方法区里面(注意:如果读过我之前的章节,就会明白这里的加载代表的是:加载->验证->准备->解析->初始化,类的所有类变量都会被赋值)。如下图
执行代码指令的程序计数器
我们知道,被加载到jvm的类对象是我们写的.java文件被编译之后的.class文件。
在编译过后会将我们的代码编译成计算机能读懂的字节码。而这个.calss文件就是,就是我们代码编译好的字节码了。
加载到内存以后,字节码执行引擎就开始工作了。去执行我们编译出来的代码指令,如下图
此时问题来了,我们是不是需要一块内存空间来记录我们字节码执行引擎目前执行到了哪行代码?这一块特殊的内存区域就是“程序计数器”
这个程序计数器就是用来记录当前执行的字节码指令的位置。
如下图:
到这里我相信会有人产生疑惑,就按照当前的代码顺序执行就行了,为什么要记录执行到哪里了?
因为我们写好的代码可能会开启多个线程并发的执行不同的代码。可能当前线程这段代码还没有执行完毕,就上下文切换到另一段代码中。
当线程再次上下文切换到之前的代码时,就需要一个专门记录当前线程执行到了哪一条字节码。所以,每一个线程都有这自己的程序计数器。
如下图:
java虚拟机栈
java代码在执行的时候,一定是某个线程来执行某个方法中的代码。
当线程执行到某个方法的时候,如果这个方法有局部变量,那么就需要一块区域来存放局部变量的数据信息。这个区域就叫做java虚拟机栈。
每一个线程都有一个自己的java虚拟机栈,比如说当执行main方法的时候就会有一个main线程,用来存放main方法中定义的局部变量
public static void main(){
People people = new People();
int i = 9;
}
比如上面的main()方法中,其实就有一个"people"的局部变量,他是引用一个People的实例对象的,这个对象我们先不管他。然后有一个"i"的局部变量。
如下图:
我想大家应该都知道栈的数据结构,后进先出。当方法执行完毕以后,这个栈桢就会出栈,里面的局部变量信息就会从内存删除。所以局部变量是线程安全的。因为只有当前线程能获取到这个值。
为什么要用后进先出的数据结构?
假设a方法当中同步调用b方法,此时a方法的栈桢先入栈,然后再是b方法的栈桢入栈。b方法执行完毕后,b方法的栈桢出栈,继续执行a方法。所以使用一个后进先出的栈结构是非常完美的。
此时jvm的内存模型图如下:
java堆内存
这一块内存是非常非常重要的。
我们实例化的所有对象都是存放在这个内存中。这个实例化的对象里面会包含一些数据,我们用上面的代码来做栗子。
public class Student{
private String name = "lisi";
public String getNmae(){
return name;
}
}
public class People{
public static void main(){
Student student = new Student();
student.getName();
}
}
还是这个代码,当main线程执行main()方法的时候,首先在堆内存中实例化Student对象,然后在局部变量中创建student,student存的是实例化Student对象的内存地址。然后执行Student对象的getName()方法。
如下图:(图中有个问题,people应该全部修正为student)
由上图可以看出来,栈空间是封闭的,是线程安全的,而堆内存中是我们主要发生线程不安全的地方,因为堆内存的空间所有的线程其实都是能共享的。
此时jvm的内存划分的最终模型为:
其他内存区域
很多java程序猿对这一块区域的接触是非常少的。
其实在JDK的很多底层代码API中,比如NIO。
如果你去看源码会发现很多地方的代码不是java写的,而是走的native方法去调用本地操作系统里面的一些方法,可能调用的都是c语言写的方法。
比如说:public native int hashCode();
在调用这种native方法的时候,就会有线程对应的本地方法栈,这个其实类似于java虚拟机栈。也是存放各种native方法的局部变量表之类的信息。
还有一块区域,是不是jvm的,通过NIO中的allocateDirect这种API,可以在jva堆外分配内存空间,然后通过java虚拟机栈里的DirectByteBuffer来引用和操作堆外内存空间。
总结
基本上jvm的核心内存区域的功能都解释清楚了,面试能回答到这一个地步应该也能顺利通过了。
我们需要重点关注的是方法区,程序计数器,java虚拟机栈和java堆内存这些内存区域的作用。
java内存区域的划分的更多相关文章
- Java虚拟机--Java内存区域的划分和异常
Java内存区域的划分和异常 运行时数据区域 JVM在运行Java程序时候会将内存划分为若干个不同的数据区域. 程序计数器 线程私有.可看作是当前线程所执行的字节码的行号指示器,字节码解释器的工作是通 ...
- Java内存区域的划分和异常
Java内存区域的划分和异常 运行时数据区域 JVM在运行Java程序时候会将内存划分为若干个不同的数据区域. 打开百度App,看更多美图 程序计数器 线程私有.可看作是当前线程所执行的字节码的行 ...
- 【转】Java内存管理:深入Java内存区域
转自:http://www.cnblogs.com/gw811/archive/2012/10/18/2730117.html 本文引用自:深入理解Java虚拟机的第2章内容 Java与C++之间有一 ...
- Java虚拟机2:Java内存区域及对象
几个计算机的概念 为以后写文章考虑,也为巩固自己的知识和一些基本概念,这里要理清楚几个计算机中的概念. 1.计算机存储单位 从小到大依次为位Bit.字节Byte.千字节KB.兆M.千兆GB.TB,相邻 ...
- Java内存管理:深入Java内存区域
Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的高墙,墙外面的人想进去,墙里面的人却想出来. 概述: 对于从事C和C++程序开发的开发人员来说,在内存管理领域,他们既是拥有最高权力的皇帝 ...
- 深入理解java虚拟机系列(一):java内存区域与内存溢出异常
文章主要是阅读<深入理解java虚拟机:JVM高级特性与最佳实践>第二章:Java内存区域与内存溢出异常 的一些笔记以及概括. 好了開始.假设有什么错误或者遗漏,欢迎指出. 一.概述 先上 ...
- 深入理解java虚拟机之java内存区域
java虚拟机在执行java程序的时候会把它所管理的内存分为多个不同的区域,每个区域都有不同的作用,以及由各自的生命周期,有些随着虚拟机进行的启动而存在,有些区域则依赖于用户线程的启动或结束而建立或销 ...
- 《深入理解Java虚拟机》-----第2章 Java内存区域与内存溢出异常
2.1 概述 对于从事C.C++程序开发的开发人员来说,在内存管理领域,他们即是拥有最高权力的皇帝又是执行最基础工作的劳动人民——拥有每一个对象的“所有权”,又担负着每一个对象生命开始到终结的维护责任 ...
- Java内存管理(一):深入Java内存区域
本文转自:http://www.cnblogs.com/gw811/archive/2012/10/18/2730117.html#undefined 推荐查看原文,原文格式更好一些. 本文引用自:深 ...
随机推荐
- Codeforces Round #676 (Div. 2) XORwice、Putting Bricks in the Wall、Palindromifier
题目链接:XORwice 题意:给你两个数a.b.求一个数x,使得((a异或x)+(b异或x))这个值最小,输出最小那个x 题解: 输出(a|b)-(a&b)就行(猜了一手 代码: #incl ...
- 【noi 2.6_9280】&【bzoj 1089】严格n元树(DP+高精度+重载运算符)
题意:定义一棵树的所有非叶节点都恰好有n个儿子为严格n元树.问深度为d的严格n元树数目. 解法:f[i]表示深度为<=i的严格n元树数目.f[i]-f[i-1]表示深度为i的严格n元树数目.f[ ...
- Warm up HDU - 4612 树的直径
题意:给出n个点和m条边的无向图,存在重边,问加一条边以后,剩下的桥的数量最少为多少. 题解: 你把这个无向图缩点后会得到一个只由桥来连接的图(可以说这个图中的所有边都是桥,相当于一棵树),然后我们只 ...
- 后缀数组Da模板+注释 以及 dc3模板
后缀数组Da模板: 1 /* 2 后缀数组倍增法Da板子 3 */ 4 #include <cstdlib> 5 #include <cstring> 6 #include & ...
- 一、Python简介及下载安装
一.关于Python Python是目前比较受欢迎的脚本语言之一,具有简洁性.易读性以及可扩展性的特点. Python与Java均可以写网页,也可以写后台功能,区别是Python执行效率低,开发效率高 ...
- office响应慢,但电脑速度没问题的解决方案
看了非常多教程都没有用,有点崩,最后终于解决了,记录一下. 问题描述 :office启动没问题,但word打开后,选中一段文字非常慢,大概延迟鼠标移动2-3秒.点击工具栏时也有延迟(点击动画看的十分清 ...
- MySQL 事务日志
重做日志(Redo log) 重做日志(Redo log),也叫做前滚日志,存放在如下位置,轮询使用,记录着内存中数据页的变化,在事务 ACID 过程中,主要实现的是 D(Durability)的作用 ...
- LeetCode 856. Score of Parentheses 括号的分数
其实是这道题的变式(某港带同学的C/C++作业) 增加一点难度,输入的S不一定为平衡的,需要自己判断是否平衡,若不平衡输出为0. 题目描述 Given a parentheses string s, ...
- hdu-6237
A Simple Stone Game Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 262144/262144 K (Java/Ot ...
- ASP.NET Core 中间件(Middleware)(一)
本文主要目标:记录Middleware的运行原理流程,并绘制流程图. 目录结构: 1.运行环境 2.Demo实践 3.源码追踪 4.AspnetCore内置middleware 一.运行环境 Visu ...