当前位置:首页 > > 架构师社区
[导读]学过 JVM的人都知道在JVM中在执行Java程序的过程中会把JVM管理的内存进行划分,叫做 运行时数据区。 JVM中管理的内存主要分为以下五个部分: 方法区(Method Area): Java堆(Heap):Java堆是JVM管理内存中最大的一块区域,几乎所有的Java对象的内存都在这


学过 JVM的人都知道在JVM中在执行Java程序的过程中会把JVM管理的内存进行划分,叫做 运行时数据区

JVM中管理的内存主要分为以下五个部分:

  1. 方法区Method Area):

  2. Java堆Heap):Java堆是JVM管理内存中最大的一块区域,几乎所有的Java对象的内存都在这里分配,此区域也是GC最活跃的区域。

  3. 虚拟机栈VM Stack):此区域就是我们通常所说的“”,当线程执行方法时会在此区域创建一个栈帧,用于储存局部变量表动态链接方法出口等信息。

  4. 本地方法栈Native Method Stack):本地方法栈虚拟机栈类似,只不过本地方法栈执行的是Native方法服务。

  5. 程序计数器Program Counter Register):是当前线程执行字节码的行号指令器,是线程私有的。

当执行一个Java方法,程序计数器记录的是JVM正在执行的字节码指令,若是执行的是本地方法,程序计数器为空。

你的对象在哪里?长什么样?我带你去看一看

这五个部分都有自己的功能,有些区域是线程私有的,会跟随着线程的创建和销毁,有些区域是跟随进程的启动而存在,各自承担着自己的职责。

在方法区中还有一块区域就是常量池,很多人叫它为永生代 PermGen(JDK 1.7的说法)。

随着JDK8的到来,JVM不再有PermGen。但类的元数据信息还在,不再是存储在连续的堆空间上,而是移动到叫做Metaspace(元空间)的本地内存(Native memory)中。

我们知道当我们执行创建对象的时候,就会初始化对象的属性信息,例如执行如下的代码:

public class TestObj {

    private int n=1;

    public static void main(String[] args) throws Exception {

        TestObj obj =new TestObj();

    }
}

该代码非常的简单,在main方法中执行TestObj obj =new TestObj()就会初始化该对象的成员变量n为1,我们都知道这个过程叫做初始化,在初始化的时候也会进行半始化。

那么什么叫做半初始化呢?

半初始化就是当一个成员变量还未初始化为它真正的值,会先初始化为它默认是的值,例如int会先初始化为0boolean会先初始化为false等。

对于半初始化,真正的讲清楚要从执行的字节码指令分析,下面我们通过字节码指令进行深入的分析,一个对象的初始化过程。

具体在idea中查看字节码指令的方法可以自行百度,这个不难,通过idea中可以查看上面代码执行的指令,如下所示:

你的对象在哪里?长什么样?我带你去看一看

在上面mian方法中执行完,也就是对应这五条指令,如下所示:

  1. new指令:表示首先在堆中申请一块内存,此时堆中的内存中存储着该对象属性n的半初始化状态值n=0。

  2. dup指令:表示复制引用。

  3. invokespecial指令:表示调用对象的初始化方法,后面对应的注释Method " " ,此时属性值n才会被初始化为1。

  4. astore_1指令:此时会将TestObj obj =new TestObj()的引用obj 与该堆中的对象建立连接。

  5. return指令:执行完最后返回。

从上面的指令中分析可以看出,当创建一个对象的时候,主要分为以下三个步骤,执行的原理图如下:

你的对象在哪里?长什么样?我带你去看一看

了解完对象的半初始化,那么什么又是对象分配?

说到JVM中的对象分配,我们得从对象在JVM中执行new指令后开始讲起。客观且慢,请听我详细道来。

在JVM中当遇到一条new指令时,会首先检查这条指令的参数是否在常量池中能定位到一个类的符号引用,若是定位不到,就表示没有被加载、解析和初始化过,就会先执行加载该类。

JVM中加载类信息的详细过程,请参考这一篇文章[面试官:你知道java类是怎么跑起来的吗?问的我一脸懵]。

若是存在该符号引用表示之前已经加载过该类信息,接下来就直接执行在堆中进行对象内存的分配。

但是随着JVM的发展,JIT编译器的出现,所有的对象分配在堆中就不那么绝对了,当创建对象为对象分配内存时,也会尝试在栈上分配,在JVM书籍中的描述如下所示:

你的对象在哪里?长什么样?我带你去看一看

那么什么是逃逸技术? 每个线程执行方法都会创建一个栈帧,该栈帧用于存储方法的局部变量,当一个变量不会在其他方法中使用到,只在该方法中使用,就不会逃逸。

什么又是变量替换呢? 标量替换就是创建一个对象的时候,直接以对象的属性进行入栈存储,方法结束后直接弹栈结束,不会有GC的介入。

因此,在栈上分配是对JVM的一种优化措施,减少了GC的活动,提高了Java虚拟机的执行效率。

当对象执行在堆上进行内存分配的时候,为了防止多线程分配内存存在混乱的情况,通常在多线程的时候对对象内存的分配    有以下两种方案进行解决:

  1. 对分配内存的动作进行同步,但是同步的的操作太消耗性能,大大降低了JVM的性能。

  2. 对堆内存为每一个线程划分一块本地线程分配缓冲TLAB),是线程私有的,这样每一个线程只需要在自己的TLAB中进行分配即可,就不用进行同步,也能达到线程安全的目的:

那么当一个对象在堆中分配完一个内存后,对象在堆中又是怎么存在的呢?

客观不急请听我慢慢道来,当对象在堆中进行完内存分配后,一个普通对象在堆中以如下图的形式存在:

你的对象在哪里?长什么样?我带你去看一看
  1. markword:存储的是锁信息、对象的年龄信息等。

  2. class pointer:类型指针指向该对象class类型。

  3. instance data:实例数据,储存对象实例的数据值。

  4. padding:对其若是该对象的大小不能被8字节整除,就会自动补齐为8字节的整数倍。

那么对象都已经存在堆中了,我们又是怎么访问该对象的?

若要访问堆中已经存在的对象,有以下两种方式:

(1) 句柄的方式:会在堆中划分一块下的内存作为句柄池,对象的引用不会直接存储数据的地址,而是指向句柄池的指针,由句柄池的指针存储数据的地址。

句柄池的方式,由于对象引用不会直接指向数据的地址,这样当GC进行回收垃圾的时候,移动对象,对象的地址改变了就不用改变reference的本身内容。

这个也是句柄访问方式的唯一优点,具体句柄访问方式的原理图如下所示:

你的对象在哪里?长什么样?我带你去看一看

(2) 直接方式:直接方式是reference直接指向数据的,这样减少了一次指针的定位,速度快,直接访问的方式原理图如下:

你的对象在哪里?长什么样?我带你去看一看

注意: 在HostSpot的源码实现中,使用的是第二种直接访问的方式

特别推荐一个分享架构+算法的优质内容,还没关注的小伙伴,可以长按关注一下:

你的对象在哪里?长什么样?我带你去看一看

长按订阅更多精彩▼

你的对象在哪里?长什么样?我带你去看一看

如有收获,点个在看,诚挚感谢

免责声明:本文内容由21ic获得授权后发布,版权归原作者所有,本平台仅提供信息存储服务。文章仅代表作者个人观点,不代表本平台立场,如有问题,请联系我们,谢谢!

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭