JVM笔记【1】-- 运行时数据区
(一)java内存区域管理
C/C++每一个new操作都需要自己去delete/free,而java里面有虚拟机自动管理内存,不容易出现内存泄漏或者溢出的问题,但是不容易出现不代表不出现,了解虚拟机怎么使用和管理内存是十分重要的是,对程序优化或者问题排查有帮助。
运行时区域主要分为:
- 线程私有:
- 程序计数器:
Program Count Register
,线程私有,没有垃圾回收 - 虚拟机栈:
VM Stack
,线程私有,没有垃圾回收 - 本地方法栈:
Native Method Stack
,线程私有,没有垃圾回收
- 程序计数器:
- 线程共享:
- 方法区:
Method Area
,以HotSpot
为例,JDK1.8
后元空间取代方法区,有垃圾回收。 - 堆:
Heap
,垃圾回收最重要的地方。
- 方法区:
1.1 程序计数器
空间很小,当前线程执行的字节码的行号指示器(线程独有,指示当前执行到哪,下一步需要执行哪一个字节码),分支,循环,跳转,异常处理,线程恢复都需要依赖它。
线程私有:java
多线程其实是线程轮流切换并分配处理器执行时间的方式实现,一个核一个具体的时间点,只会执行一个线程的指令。线程切换需要保存和恢复正确的执行位置(保护和恢复现场),所以不同的线程需要不同的程序计数器。
- 执行
java
方法时,程序计数器记录的是正在执行的字节码指令地址 - 执行
Native
方法,程序计数器为空
唯一一个没有规定任何OutOfMemory
的区域,也没有GC(垃圾回收)。
1.2 虚拟机栈
线程私有,生命周期和线程一样,主要是记录该线程Java方法执行的内存模型。虚拟机栈里面放着好多栈帧。注意虚拟机栈,对应是Java方法,不包括本地方法。
一个Java方法执行会创建一个栈帧,一个栈帧主要存储:
- 局部变量表
- 操作数栈
- 动态链接
- 方法出口
每一个方法调用的时候,就相当于将一个栈帧放到虚拟机栈中(入栈),方法执行完成的时候,就是对应着将该栈帧从虚拟机栈中弹出(出栈)。
每一个线程有一个自己的虚拟机栈,这样就不会混起来,如果不是线程独立的话,会造成调用混乱。
大家平时说的java内存分为堆和栈,其实就是为了简便的不太严谨的说法,他们说的栈一般是指虚拟机栈,或者虚拟机栈里面的局部变量表。
局部变量表一般存放着以下数据:
- 基本数据类型(
boolean
,byte
,char
,short
,int
,float
,long
,double
) - 对象引用(reference类型,不一定是对象本身,可能是一个对象起始地址的引用指针,或者一个代表对象的句柄,或者与对象相关的位置)
- returAddress(指向了一条字节码指令的地址)
局部变量表内存大小编译期间确定,运行期间不会变化。空间衡量我们叫Slot(局部变量空间)。64位的long和double会占用2个Slot,其他的数据类型占用1个Slot。
异常:
- StackOverflowError:线程请求的栈深度大于虚拟机允许的深度
- OutOfMemoryError:内存不足
1.3 本地方法栈
和虚拟机栈类似,对应本地方法,Native
,虚拟机规范允许语言,使用方式和数据结构不同,有些可能将虚拟机栈和本地方法栈合并。
异常与虚拟机栈一致:
- StackOverflowError:线程请求的栈深度大于虚拟机允许的深度
- OutOfMemoryError:内存不足
1.4 java堆
堆是内存管理最大的一块,线程共享。
虚拟机规范中说,所有的对象实例和数组都要在堆上分配。但是实际上不是所有的对象都在堆上分配,这个和JIT编译器的发展和逃逸分析技术相关。Why?
// TODO
堆的细分:新生代,老年代,再细分有Eden,From survivor,To survivor等。
堆中也有可能有线程私有的区域,分配缓冲区。
物理上可以不连续,但是逻辑上是连续的。
异常:
- OutOfMemoryError:内存不足
1.5 方法区
名为非堆,但是实际和堆一样,是线程共享的区域,主要存贮以下信息:
- 已被虚拟机加载的类信息
- 常量
- 静态变量
- 即时编译器编译后的代码
方法区不等于永久代,指示Hotspot虚拟机将GC分代收集拓展到方法区,也就是用永久代实现了方法区,而其他的虚拟机不一定,不是固定的。JDK1.7将永久代的字符串常量移出了。
方法区回收垃圾的效果不是很好,可以选择不回收,虚拟机可以决定,当然也可能发生内存泄漏。
异常:
- OutOfMemoryError:内存分配异常
1.5.1 运行时常量池
运行时常量池时方法区的一部分,但是不是全部,Class
文件主要包括:
- 类的版本
- 字段
- 方法
- 接口
- 常量池,存放编译产生的字面量和符号引用,一般除了描述Class文件的符号引用,还有直接引用也在里面。是动态的,运行时可以产生,比如String.intern()方法。
异常:
- OutOfMemoryError:内存分配异常
(二)直接内存
不是虚拟机运行时数据区,也不是规范规定的区域,但是使用频繁且可能会有OutOfMemoryError:内存分配异常出现。
比如,NIO(1.4)基于Channel与Buffer的I/O,可以用Native函数直接分配堆外内存,通过存储在Java堆中的DirectByteBuffer对象作为引用来操作,提高性能,不需要Java堆和Native堆都来回复制数据。
直接内存受物理的内存,或者处理器寻址空间之类的限制。
本文系JVM学习相关笔记,整理来自周志明老师的《深入理解Java虚拟机》,无比钦佩,强烈推荐!
【作者简介】:
秦怀,公众号【秦怀杂货店】作者,技术之路不在一时,山高水长,纵使缓慢,驰而不息。这个世界希望一切都很快,更快,但是我希望自己能走好每一步,写好每一篇文章,期待和你们一起交流。
此文章仅代表自己(本菜鸟)学习积累记录,或者学习笔记,如有侵权,请联系作者核实删除。人无完人,文章也一样,文笔稚嫩,在下不才,勿喷,如果有错误之处,还望指出,感激不尽~
JVM笔记【1】-- 运行时数据区的更多相关文章
- 1、JVM 内存模型+运行时数据区+JVM参数
JMM(内存模型) 1.’主内存+每个线程有自己的内存 JVM运行时数据区 包含:1.程序计算器(每个线程自带):2.JAVA-STACK(每个线程自带):3.本地方法stack:4.堆:5.方法区 ...
- jvm内存模型(运行时数据区)
运行时数据区(runtime data area) jvm定义了几个运行时数据区,这些运行时数据区存储的数据,供开发者的应用或者jvm本身使用.按线程共享与否可以分为线程间共享和线程间独立. 线程间独 ...
- JVM三部曲之运行时数据区 (第一部)
在接下来的几天想总结下,JVM相关的一些内容,比如下面的这三个内容算是比较核心知识点了 1.运行时数据区域: 在运行时数据区里存储类Class文件元数据(方法区),对象和数组(堆),方法参数局部变量( ...
- 《深入理解Java虚拟机》笔记01 -- 运行时数据区
运行时数据区示意图 1. 程序计数器 占用一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器.主要用来记录线程执行到哪条语句了,分支.循环.跳转.异常处理.线程恢复等功能都需要依赖这个 ...
- JVM内存结构——运行时数据区
在Java虚拟机规范中将Java运行时数据划分为6种,分别为: PC寄存器(程序计数器) Java栈 堆 方法区 运行时常量池 本地方法栈 一.PC寄存器(程序计数器) PC寄存器(Program C ...
- JVM之Java运行时数据区(线程隔离区)
来源 JVM会在会在执行Java程序过程中把所管理的内存划分为若干区域,主要包括程序计数器(Program Counter Register),虚拟机栈(VM Stack),本地方法栈(Native ...
- JVM之Java运行时数据区(线程共享区)
JVM运行时区域各线程共享的区域包括堆区和方法区. 堆区 堆区最最主要的功能是存储对象实例[上篇也提到过],因此Java垃圾回收的主要战场就是在堆区,因此也有称为GC堆区.如果堆区的内存不够会出现Ou ...
- 【转】Java运行时数据区简介及堆与栈的区别
理解JVM运行时的数据区是Java编程中的进阶部分.我们在开发中都遇到过一个很头疼的问题就是OutOfMemoryError(内存溢出错误),但是如果我们了解JVM的内部实现和其运行时的数据区的工作机 ...
- JVM学习笔记-运行时数据区
不同于C,C++程序,Java程序的内存管理工作由Java虚拟机(JVM)接管,这减低了java程序员的负担,但如果出现内存泄露与溢出问题如报OutOfMemory,StackOverFlow异常错误 ...
随机推荐
- Linux初学学习笔记 -----正则表达式和通配符
简单来说通配符是用来匹配文件名和目录而正则表达式是用来匹配文本内容的 常用的通配符 *:匹配任意多个字符 下面的是以p为开头的目录里面的文件 ?:匹配任意一个字符 [-]:匹配括号内出现的任意一个字符 ...
- 安全的字符串拷贝strcpy_s的实现与理解
在C标准库中提供了字符串拷贝函数strcpy,而微软则为为它提供了一个更安全的版本strcpy_s,其函数原型为 errno_t __cdecl strcpy_s( char* _Destinatio ...
- 关于transition动画效果中,滚动条会闪一下就消失的问题
具体问题说明: 我在通过transition来改变width的长度,在transition变化过程中,底下的滚动条会闪烁一下. 问题原理:因为是里面容器没办法完全被装下,并且容器的宽度被限制住了. 解 ...
- linux 权限提升
1.内核提权,根据版本搜索相应exp 查看操作系统版本命令 uname –a lsb_release –a cat /proc/version 查看内核版本 cat /etc/issue 查看发行类型 ...
- 使用文件描述符作为Python内置函数open的file实参调用示例
一.关于文件描述符 open()函数的file参数,除了可以接受字符串路径外,还可以接受文件描述符(file descriptor),文件描述符是个整数,对应程序中已经打开的文件. 文件描述符是操作系 ...
- PyQt(Python+Qt)学习随笔:QAbstractItemView的editTriggers属性以及平台编辑键(platform edit key )
老猿Python博文目录 老猿Python博客地址 editTriggers属性 editTriggers属性用于确认哪些用户操作行为会触发ItemView中的数据项进入编辑模式. 此属性是由枚举类E ...
- Linux(宝塔)部署.Net Core完整记录
前言 最近在V站上看到一个外卖推广的小程序,意思大概是类似淘宝联盟那种,别人走自己的链接后,自己可以抽取大概4%-6%的提成.觉得还蛮有意思的,一开始开源的是静态页面写死的,所以我这边用.Net Co ...
- REHの个人主页
朝暮与年岁并往 然后与你一同行至天光. 简介 这是怎么做到的啊-- 把那些迷茫的浑浊的不可预知的,裁剪,变化,像个造物主一样,最终成为混沌而又分明的,除去一身的戾气和险恶,把那些复杂和晦涩都剖析成它精 ...
- 使用plesk遇到的问题
按照plesk使用指南中,"快速建站"的部分,配置一番后,还是访问不了网站. 后来解决了,原因如下: 主域名没有解析,只解析了,带www的子域名 80端口没开
- Linux 开机启动程序的顺序
1.加载BISO的硬件信息,并取得第一个开机代号 2.读取第一个开机装置的mbr的boot loader的信息 3.加载kernel操作系统核心信息,开始解压缩,并驱动所有硬件装置 4.kernel执 ...