面试集中营—JVM篇

一、JVM内存模型

        线程独占:栈,本地方法栈,程序计数器;

        线程共享:堆,方法区

        虚拟机栈:线程私有的,线程执行方法是会创建一个栈阵,用来存储局部变量表,操作栈,动态链接,方法出口等信息.调用方法时执行入栈,方法返回式执行出栈;

        本地方法栈:与虚拟机栈类似,也是用来保存执行方法的信息.执行Java方法是使用栈,执行Native方法时使用本地方法栈;

        程序计数器:保存着当前线程执行的字节码位置,每个线程工作时都有独立的计数器,只为执行Java方法服务,执行Native方法时,程序计数器为空;

        堆:内存管理最大的一块,对被线程共享,目的是存放对象的实例,几乎所有的对象实例都会放在堆中;

        方法区:称非堆区,用于存储已被虚拟机加载的类信息,常量,静态变量,即时编译器优化后的代码等数据.1.7的永久栈和1.8的元空间都是方法区的一种实现;

        除此以外,还有一块儿地方叫做堆外内存:Direct ByteBuffers

二、类加载与卸载

类加载的过程

        三步走:加载-链接-初始化;

        五步走:加载-验证-准备-分析-初始化

        不管是几步走,我们可以这么理解,我写了一段加密程序,然后我给了你一个解密程序。现在我把加密程序给你,为了能够把我的加密程序变成你能够使用的程序(比如a+b=c),那么第一步应该是把加密程序先下载并打开(加载)— 检查一下这个加密程序是否符合解密规范且不会对解密程序造成危害(验证)— 把加密程序变成可执行文件(准备、分析、初始化)

  • 加载通过类的完全限定名,查找此类字节码文件,利用字节码文件创建Class对象.
  • 验证确保Class文件符合当前虚拟机的要求,不会危害到虚拟机自身安全.
  • 准备进行内存分配,为static修饰的类变量分配内存,并设置初始值(0null).不包含final修饰的静态变量,因为final变量在编译时分配.
  • 解析将常量池中的符号引用替换为直接引用的过程.直接引用为直接指向目标的指针或者相对偏移量等.
  • 初始化主要完成静态块执行以及静态变量的赋值.先初始化父类,再初始化当前类.只有对类主动使用时才会初始化.

 加载机制-双亲委派机制

        这里的双亲不是真的有两个类加载器,一个父亲一个母亲,而是指parent,parent就是父母的意思,翻译过来叫双亲;

        即加载器加载类时先把请求委托给自己的父类加载器执行,直到顶层的启动类加载器。父类加载器能够完成加载则成功返回,不能则子类加载器才自己尝试加载。换句话说遇到class加载先叫爸爸,爸爸找自己的爸爸,自己的爹搞不定了自己再尝试加载。

        优点就是避免类的重复加载、避免Java的核心API被篡改,比如你写了一个String类,难道系统启动的时候要使用我们自己的写的String类来替换java.lang下的String类吗?

类加载器

        类的加载是由类加载器完成的,类加载器包括:根加载器(BootStrap)、扩展加载器 (Extension)、系统加载器(System)和用户自定义类加载器(java.lang.ClassLoader的子 类)。从Java2JDK1.2)开始,类加载过程采取了父亲委托机制(PDM)。PDM更好地保证了Java平台的安全性,在该机制中,JVM自带的Bootstrap是根加载器,其他的加载器都有且仅有一个父类加载器。类的加载首先请求父类加载器加载,父类加载器无能为力时才由其子类加载器自行加载。JVM不会向Java程序提供对Bootstrap的引用。

        Bootstrap:一般用本地代码实现,负责加载JVM基础核心类库(rt.jar);

        Extension:从java.ext.dirs系统属性所指定的目录中加载类库,它的父加载器是Bootstrap

        System:又叫应用类加载器(Application),其父类是Extension。它是应用最广泛的类加载器。它从环境变量classpath或者系统属性java.class.path所指定的目录中记载类,是用户自定义加载器的默认父加载器;

三、对象分配规则

        这里主要讲述对象在堆内存中的生命历程。

        1、对象优先分配在Eden区,当Eden区没有足够的空间时,会触发一次minorGC;

        2、大对象直接进入老年代(大对象是指需要大量连续内存空间的对象)。这样做的目的是避 免在Eden区和两个Survivor区之间发生大量的内存拷贝

        3、长期存活的对象进入老年代,老年代就是要上了一定年纪,每次minorGC,还存活的对象年龄加1,年龄到达某个值(默认好像是15)就会进入老年代,表示这个对象应该会长期活在内存中,把Eden区的空间留给新的对象;

        4、动态判断对象的年龄。如果Survivor区中相同年龄的所有对象大小的总和大于Survivor 空间的一半,年龄大于或等于该年龄的对象可以直接进入老年代;

        5、空间分配担保:每次进行MinorGC时,JVM会计算Survivor区移至老年区的对象的平均大小,如果这个值大于老年区的剩余值大小则进行一次FullGC;老年代无法分配这么多的内存可能会造成内存溢出;

四、Java对象的创建过程

         1、当遇到一条新建对象的指令时,先要检查能否在常量池中找到定义这个类的符号引用,如果存在引用,就加载这个类;

         2、为对象分配内存空间:

        指针碰撞:如果内存分配是一片连续的区域,那么只需要把空闲指针移动内存大小即可分配内存;

        空闲列表:如果内存分配是不连续的,就需要一个列表来记录哪些是空闲的; 

        TLAB:本地线程分配缓冲区(Thread Local Allocation Buffer)是Java虚拟机堆内存中的一个特殊区域,专门为线程分配对象而设计。TLAB是Eden区的一部分,每个线程在初始化时会被分配一块TLAB空间,这块空间仅供当前线程使用。线程在分配内存时,会在自己的TLAB上进行分配,这样可以避免多线程间的内存分配竞争,从而提高分配效率。

        3、将除对象头外的对象内存空间初始化为0

        4、初始化对象头信息

五、Java的对象结构

        对象由三个部分组成:对象头、实例数据、对齐填充

        对象头:由两部分组成,第一部分存储对象自身的运行时数据:哈希码、GC分代年龄、锁标识状态、线程持有的锁、偏向线程ID(一般占32/64bit)。第二部分是指针类型,指向对象的类元数据类型(即对象代表哪个类)。如果是数组对象,则对象头中还有一部分用来记录数组长度。

        实例数据:用来存储对象真正的有效信息(包括父类继承下来的和自己定义的);
        
        对齐填充:JVM 要求对象起始地址必须是 8 字节的整数倍( 8 字节对齐)

六、逃逸分析技术

        逃逸分析(EscapeAnalysis),是一种可以有效减少Java程序中同步负载和内存堆分配压力的跨函数全局数据流分析算法。通过逃逸分析,JavaHotspot编译器能够分析出一个新的对象的引用的使用范围,从而决定是否要将这个对象分配到堆上。

        通俗点讲,如果一个对象的指针被多个方法或者线程引用时,那么我们就称这个对象的指针发生了逃逸。优点如下:

  • 栈上分配,可以降低垃圾收集器运行的频率;这样的好处有,一、减少内存使用,因为不用生成对象头。二、程序内存回收效率高,并且GC频率也会减少
  • 同步消除,如果发现某个对象只能从一个线程可访问,那么在这个对象上的操作可以不需要同步;

七、为什么使用元空间替换了永久栈

        先看几个概念:

        方法区: 方法区和堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译后的代码等数据;

        永久代:在JDK1.7及以前的版本中,永久代就是方法区在Hotspots中的实现

        元空间:对于Java8HotSpots取消了永久代,取而代之的是元空间(Metaspace)。换句话说,就是方法区还是在的,只是实现变了,从永久代变为元空间了。

        那么为什么要替换呢?主要是解决方法区内存OOM的问题,方法区的内存和堆内存是一片连续的空间,默认最大值是64MB,而1.8以后的方法区放在独立的元空间中,和堆内存是不连续的,没有内存的限制,理论上系统内存剩余多大就可以用多大;

八、STW、Oopmap与安全点

        STW:Stop the world, 在进行垃圾回收的时候,当需要移动对象时为了保证对象引用更新的正确性,必须要停止所有的用户线程,等移动结束以后再放开,这就是STW;

        Oopmap:HotSpot中,有个数据结构(映射表)称为Oopmap,HotSpot会把对象内什么偏移量上是什么类型的数据计算出来,记录到OopMap。在即时编译过程中,也会在「特定的位置」生成OopMap,记录下栈上和寄存器里哪些位置是引用。 这些特定的位置就叫安全点

        安全点:Safe  Point,主要在以下三种地方

        1、循环的末尾; 

        2、方法临返回前/调用方法的call指令后;

        3、可能抛异常的位置;

        STW并不是任意时刻都可以进行,用户线程要停顿下来开始垃圾收集,必须是执行到安全点才能够暂停;

       

        

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/324275.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Vue3进阶】- Pinia

什么是Pinia Pinia 是 Vue 的专属状态管理库,它允许你跨组件或页面共享状态。它提供了类似于 Vuex 的功能,但比 Vuex 更加简单和直观。 需要在多个组件之间共享状态或数据时使用 Pinia 的 store,这样可以避免 props 和 eventBus 等传统方法…

LeetCode:116.填充每个节点的下一个右侧节点指针

文章目录 1.层次遍历2.使用next层序遍历3.递归方法 LeetCode:116.填充每个节点的下一个右侧节点指针 题目: 示例: 分析题意容易关注到只需要将每层结点连接起来,因此我们只需要把每层结点求出来即可,即使用层次遍历。 …

面试中算法(删去n个数字后的最小值)

有一个整数,从该整数中去掉n个数字,要求剩下的数字形成的新整数尽可能小。 分析:使用栈的特性,在遍历原整数的数字时,让所有数字一个一个入栈,当某个数字需要被删除时,(即栈顶数字&g…

【深度学习】Diffusion扩散模型原理解析1

1、前言 diffusion,这几年一直很火的模型,比如这段时间在网上的文生图大模型——Stable diffusion。就是以diffusion作为基底模型,但由于该模型与VAE那边,都涉及了较多了概率论知识,实在让人望而却步。所以&#xff0…

node pnpm修改默认包的存储路径

pnpm与npm的区别 PNPM和NPM是两个不同的包管理工具。 NPM(Node Package Manager)是Node.js的官方包管理工具,用于安装、发布和管理Node.js模块。NPM将包安装在项目的node_modules目录中,每个包都有自己的依赖树。 PNPM&#xf…

NumPy及Matplotlib基本用法

NumPy及Matplotlib基本用法 导语NumPy导入与生成算术运算N维数组广播元素访问 Matplotlib简单图案绘制多函数绘制图像显示 总结参考文献 导语 深度学习中经常需要对图像和矩阵进行操作,好在python提供了Numpy和Matplotlib库,前者类似一个已经定义的数组…

2024年数维杯B题完整代码和思路论文讲解与分析

2024数维杯数学建模完整代码和成品论文已更新,获取↓↓↓↓↓ https://www.yuque.com/u42168770/qv6z0d/bgic2nbxs2h41pvt?singleDoc# 2024数维杯数学建模B题45页论文和代码已完成,代码为全部问题的代码 论文包括摘要、问题重述、问题分析、模型假设、…

linux phpstudy 重启命令

[rootLinuxWeb phpstudy]# ./system/phpstudyctl restart 查看命令 1) phpstudy -start 启动小皮面板 2) phpstudy -stop 停止小皮面板 3) phpstudy -restart 重启小皮面板 4) phpstudy -status 查询面板状态 5) phpstudy -in…

pytest(二):关于pytest自动化脚本编写中,初始化方式setup_class与fixture的对比

一、自动化脚本实例对比 下面是一条用例,使用pytest框架,放在一个类中,两种实现方式: 1.1 setup_class初始化方式 1. 优点: 代码结构清晰,setup_class 和 teardown_class 看起来像传统的类级别的 setup 和 teardown 方法。2. 缺点: 使用 autouse=True 的 fixture 作为…

Linux 磁盘分区工具 gdisk / fdisk

fdisk 是传统的 Linux 磁盘分区工具,磁盘容量有2T的大小限制;gdisk 又叫 GPT fdisk, 作为 fdisk 的升级版,主要使用的是GPT分区类型,用来划分容量大于2T的硬盘,本文介绍使用方法。 简介 早期的磁盘使用 fdisk 工具分区…

GitHub搭建免费博客

一、GitHub仓库准备 ​ 搭建博客需要准备两个仓库。一个存放博客图床的仓库,另一个存放博客网站的仓库。 1.1、图床创建 新建仓库 第一步: ​ 第二步: 生成Token令牌 点击右上角头像->Settings->下拉,直到左侧到底&#…

中国地图(2024版审图号地图)和地图变化说明

2024版shp格式审图号地图预览图: 新版中国地图的变化(简述) 国土面积的增加:新版中国地图显示,中国的国土面积从960万平方公里增加到1045万平方公里,增加了85万平方公里。 九段线变为十段线:…

GT资源-Clock资源

一、Transmitter 时钟分布 XCLK:在使用TX buffer的模式下,XCLK来源于TXOUTCLK。在使用TX bypassing的模式下XCLK来源于TXUSERCLK。TXUSRCLK是GTX/GTH中PCS的内部逻辑时钟。TXUSRCLK2是GT Transceiver 用户侧逻辑时钟。 TXUSRCLK与TXUSRCLK2的关系 FPGA …

无人零售,重塑购物新纪元

在这个快节奏的时代,科技的每一次跃进都在悄无声息地改变着我们的生活方式。而今,无人零售正以雷霆之势,颠覆传统购物模式,为我们带来前所未有的便捷与智能体验。想知道无人零售如何彻底改变我们的购物方式吗?跟随我&a…

字符以及字符串函数

字符以及字符串函数 求字符串长度strlen 长度不受限制的字符串函数strcpystrcatstrcmp 长度受限制的字符串函数strncpystrncatstrncmp 字符串查找strstrstrtok 错误信息报告strerror 字符分类函数字符转换函数tolowertoupper 内存操作函数memcpymemmovememcmpmemset 这篇文章注…

基于Springboot的校园生活服务平台(有报告)。Javaee项目,springboot项目。

演示视频: 基于Springboot的校园生活服务平台(有报告)。Javaee项目,springboot项目。 项目介绍: 采用M(model)V(view)C(controller)三层体系结构…

(已解决)org.springframework.amqp.rabbit.support.ListenerExecutionFailedException

报错截图 解决方案 1、登录rabbitMQ网址,删除所有队列 2、重启rabbitMQ 亲测有效!!!亲测有效!!!亲测有效!!!

简单问题汇总

一、vector和list 1.vector vector是可变大小数组的序列容器,拥有一段连续的内存空间,并且起始地址不变,因此能高效的进行随机存取,时间复杂度为o(1);但因为内存空间是连续的,所以在进行插入和删除操作时…

torch.searchsorted

torch.searchsorted 官方文档链接:torch.searchsorted — PyTorch 2.3 documentation 该函数用于在已排序的序列中查找要插入的值的位置,以保持序列的顺序, torch.searchsorted(sorted_sequence, values, *, out_int32False, rightFalse, s…

Redis是单线程吗?为什么6.0之后引入了多线程?

Redis是单线程吗?为什么6.0之后引入了多线程? Redis 是单线程吗?Redis 单线程模式是怎样的?Redis 采用单线程为什么还这么快?Redis 6.0 之前为什么使用单线程?Redis 6.0 之后为什么引入了多线程&#xff1f…