java8堆内存模型_简述JVM内存模型

3528ef78476eccce352b6a091916e559.png

我们都知道,计算机中所有程序都是再内存中运行的,在程序运行的过程中,需要不断的将内存的逻辑地址和物理地址进行映射,找到相应的命令和数据。作为操作系统进程,Java也会面临内存限制。即内存架构所提供的可寻址地址空间,在32位处理器中,这个可寻址空间位2^32可寻址范围,64可寻址空间为2^64。在操作系统中,定义了两块空间,即内核空间和用户空间,Java运行在用户空间上。

b1515380a68e5ffdb05afa7396d9a0bc.png
JVM内存模型


在JVM中,可以从线程私有和共享来划分它的内存模型

线程私有的:程序计数器,虚拟机栈,本地方法栈

共享的:方法区,Java堆

程序计数器(Program Counter Registers)

当前代码的字节码的指示器,它可以选取下一条需要执行的字节码指令,进程序的流程控制,包括循环,if分支等等。程序计数器在只占还很小的一块内存空间,它线程私有的,在确定的时刻,一个线程都会在一个处理器上执行,为了保证线程切换后能在正确的执行位置上,所以每个线程都需要一个程序计数器。对于程数计数器,不用担心内存泄露,因为它的内存地址是逻辑地址而非物理地址。如果执行的是Native方法,计数器位是undefined。

Java虚拟机栈(VM Stack)

Java虚拟机栈也是线程私有的,每个方法执行的时候都会创建一个栈帧,它是方法运行时的基础数据结构。栈帧中主要存储以下的信息:

  • 局部变量表
  • 操作数栈
  • 动态链接
  • 地址返回

当方法调用结束的时候,栈帧会被销毁。
在局部变量表的中,存储了编译期可知的各种Java虚拟机的基本数据类型,对象引用地址,字节码指令地址等。这些类型的在局部变量表中的存储以局部变量槽(slot)来表示,除了doublelong占了两个槽,其他都是占一个。
当递归调用层级太多的时候,就会发生StackOverflowError异常。限制递归的次数,栈有固定的容量,不需要GC释放,会自动释放。

虚拟机栈可以动态扩展,当无法申请到足够的内存就会发生OOM的异常。通过设置-xss指定每个线程虚拟机栈的大小。

public void stackOOM(){
   while(true){
      new Thread(()->{
          while(true){}
      }).start();
  }
}

Window平台的JVM映射到操作系统内核上,所以运行可能会产生假死的情况,请谨慎使用。

本地方法栈(Native Method Stack)

本地方法栈和虚拟机栈类似,JVM并未要求本地方法栈一定实现某种语言的方法调用,使用者可以灵活去进行调用。

Java堆(Heap)

6fa23af55be9f727350f6dee7c141a07.png
堆模型

Java堆在JVM中是一块最大的内存,《Java虚拟机规范》描述“所有对象和数组都应该在堆上分配”,但是随着即时编译技术的发展,在Java11中,我们已经可以看到可以使用Java不经过编译直接运行了,实现了真正的“JavaScript”,这些逃逸分析,标量替换的技术可以实现对象在栈上分配。

如上图所示,在堆上划分了很多区域,Java堆实际上是垃圾收集器(GC)管理的一块内存区域,经过之上的划分得以进行更有效率的垃圾回收。在后面的文章会写到。

另外,在堆中还定义了一些设置参数。通过Xmx设置堆最大内存,Xms设置最小内存,Xms和Xmx一般都设置一样大小的,因为再扩容的时候会引起内存抖动,影响性能。

Java堆在物理上不要求连续,在逻辑上连续即可。

方法区(Method Area)

方法区是线程共享的区别,在Java8以前,方法区中定义了永久代。因为使用永久代来实现了方法区,所以被描述为堆的一个逻辑部分。但是它确是“非堆”,只是设计堆中的收集器 扩展到了方法区而已。在Java8的时候,永久代被替换成了元空间(Metaspace)。这样做的好处有以下原因:

  • Metaspace使用的是本地内存
  • 字符串常量池存在永久代中,容易出现性能问题和内存溢出。
  • 类和方法的信息难以确定,给永久代指定大小比较困难,太小容易操作永久代溢出,太大会导致老年代溢出。
  • 永久代会为GC带来不必要的复杂性。
  • 方便不同的JVM
    在元空间中,存储了虚拟机加载的类信息,常量,静态变量等。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
为什么要学JVM1、一切JAVA代码都运行在JVM之上,只有深入理解虚拟机才能写出更强大的代码,解决更深层次的问题。2、JVM是迈向高级工程师、架构师的必备技能,也是高薪、高职位的不二选择。3、同时,JVM又是各大软件公司笔试、面试的重中之重,据统计,头部的30家互利网公司,均将JVM作为笔试面试的内容之一。4、JVM内容庞大、并且复杂难学,通过视频学习是最快速的学习手段。课程介绍本课程包含11个大章节,总计102课时,无论是笔试、面试,还是日常工作,可以让您游刃有余。第1章 基础入门,从JVM是什么开始讲起,理解JDK、JRE、JVM的关系,java的编译流程和执行流程,让您轻松入门。第2章 字节码文件,深入剖析字节码文件的全部组成结构,以及javap和jbe可视化反解析工具的使用。第3章 类的加载、解释、编译,本章节带你深入理解类加载器的分类、范围、双亲委托策略,自己手写类加载器,理解字节码解释器、即时编译器、混合模式、热点代码检测、分层编译等核心知识。第4章 内存模型,本章节涵盖JVM内存模型的全部内容,程序计数器、虚拟机栈、本地方法栈、方法区、永久代、元空间等全部内容。第5章 对象模型,本章节带你深入理解对象的创建过程、内存分配的方法、让你不再稀里糊涂。第6章 GC基础,本章节是垃圾回收的入门章节,带你了解GC回收的标准是什么,什么是可达性分析、安全点、安全区,四种引用类型的使用和区别等等。第7章 GC算法与收集器,本章节是垃圾回收的重点,掌握各种垃圾回收算法,分代收集策略,7种垃圾回收器的原理和使用,垃圾回收器的组合及分代收集等。第8章 GC日志详解,各种垃圾回收器的日志都是不同的,怎么样读懂各种垃圾回收日志就是本章节的内容。第9章 性能监控与故障排除,本章节实战学习jcmd、jmx、jconsul、jvisualvm、JMC、jps、jstatd、jmap、jstack、jinfo、jprofile、jhat总计12种性能监控和故障排查工具的使用。第10章 阿里巴巴Arthas在线诊断工具,这是一个特别小惊喜,教您怎样使用当前最火热的arthas调优工具,在线诊断各种JVM问题。第11章 故障排除,本章会使用实际案例讲解单点故障、高并发和垃圾回收导致的CPU过高的问题,怎样排查和解决它们。课程资料课程附带配套项目源码2个159页高清PDF理论篇课件1份89页高清PDF实战篇课件1份Unsafe源码PDF课件1份class_stats字段说明PDF文件1份jcmd Thread.print解析说明文件1份JProfiler内存工具说明文件1份字节码可视化解析工具1份GC日志可视化工具1份命令行工具cmder 1份学习方法理论篇部分推荐每天学习2课时,可以在公交地铁上用手机进行学习。实战篇部分推荐对照视频,使用配套源码,一边练习一遍学习。课程内容较多,不要一次性学太多,而是要循序渐进,坚持学习。      

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值