• JVM面试题-JVM内存结构解析(图文详解)


    JVM内存结构

    共享 和 隔离

    线程共享区域:方法区、堆、直接内存

    线程隔离区域:虚拟机栈、本地方法栈、程序计数器

    线程共享:定义一个变量或者一个方法,多线程都可以同时访问、修改这个方法或者变量

    线程隔离:就是数据不能被多个线程同时访问,某些数据只属于一个线程

    1.程序计数器

    线程私有的。

    作用:记录线程执行到哪一步,保存的是字节码的行号,用于正在执行的字节码指令的地址

    我的理解:相当于是一个线程的进度条,记录一下当前执行到的位置,这样CPU切换了别的线程之后,当前线程就不会丢失任务的进度,下次的话就可以继续执行。

    什么时候入栈,什么时候出栈、什么时候进入循环、跳转、异常,线程恢复等等由程序计数器来进行记录。

    2.虚拟机栈

    java中的栈通常是指虚拟机栈。

    • 虚拟机栈:每个线程运行时所需要的内存

    • 每个栈是由多个栈帧组成,对应这每次方法调用是所占的内存

    • 每个线程只能有一个活动栈帧,对应着正在执行的那个方法

    虚拟机栈:线程私有的,每个方法执行的时候都会创建一一个栈帧, 用于存储局部(本地)变量表,里面存储的就是内存地址、操作数、动态链接和方法返回等信息,当线程请求的栈深度超过了虚拟机允许的最大深度时,就会抛出StackOverFlowError;

    栈内存分配越大越好吗?

    不是,因为增加栈大小,会造成每个线程的栈都变的很大,使得一定的栈空间下,能创建的线程数量会变小。

    垃圾回收是否会涉及倒虚拟机栈?

    不会;垃圾回收只会涉及到方法区和堆中,方法区和堆也会存在栈溢出的可能; 程序计数器,只记录运行下一行的地址,不存在栈溢出和垃圾回收; 虚拟机栈和本地方法栈,都是只涉及压栈和出栈,可能存在栈溢出,不存在垃圾回收.

    栈内存溢出情况
    1. 栈帧过多导致栈内存溢出,典型问题:递归调用

    2. 栈帧过大导致栈内存溢出,你不改默认值的话一般不会出现这个问题

    方法内的局部变量是否线程安全?
    1. 如果方法内局部变量没有逃离方法的作用范围,它是线程安全的

    2. 如果是局部变量引用了对象,并逃离方法的作用范围,需要考虑线程安全

    3.本地方法栈

    操作系统的本地方法,跟java中的c语言是自己封装的,操作系统不认识,的需要操作系统的本地c语言进行翻译识别,最终一步翻译。

    本地方法栈:线程私有的,保存的是本地方法的信息,当一个jvm创建的线程调用本地方法后,jvm不会在虚拟机栈中为该线程创建栈帧,而是简单的动态链接并直接调用调用操作系统提供的某些方法;

    就是java的执行,最终需要把我们的方法翻译成操作系统可识别的语言,本地方法栈就是存放翻译后的方法

    4.堆

    是线程共享的,作用:存 对象实例、数组等。内存不够的时候:抛异常OutOfMemoryError

    组成:又分为 年轻代 和 老年代

    年轻代:被划分为三部分,Eden区和两个大小严格相同的Survivor区,根据JVM的策略,在经过几次垃圾收集后,任然存活于Survivor的对象将被移动到老年代区间。 ​ 老年代:主要保存生命周期长的对象,一般是一些老的对象

    java7和java8的 堆 区别:

    Java8为了让堆节省空间,防止内存溢出,做了优化:把 方法区/永久代 放到了本地内存中。

    • 1.7中有有一个永久代,存储的是类信息、静态变量、常量、编译后的代码

    • 1.8移除了永久代,把数据存储到了本地内存的元空间中,防止内存溢出

    5.方法区

    线程共享的。

    主要存储类的信息、运行时常量池。

    方法区:存放已被加载的类信息、常量、静态变量、即时编译器编译后的代码数据,即永久代

    类加载:JDK编译java文件后的class文件从磁盘加载到内存中,类一开始从磁盘加载到内存的时候先加载到方法区

    Class文件中除了有类的版本、字段、方法、接口等描述信息

    jdk1.8中不存在方法区了,被元数据区(元空间)替代了,原方法区被分成两部分:

    1: 加载的类信息,2:运行时常量池;

    加载的类信息被保存在元数据区中,运行时常量池保存在堆中;

    字符串常量池

    1、在JDK1.7前,运行时常量池+字符串常量池是存放在方法区中,HotSpot VM对方法区的实现称为永久代。

    2、在JDK1.7中,字符串常量池从方法区移到堆中,运行时常量池保留在方法区中。

    3、在JDK1.8中,HotSpot移除永久代,使用元空间代替,此时字符串常量池保留在堆中,运行时常量池保留在方法区中,只是实现不一样了,JVM内存变成了直接内存。

    运行时常量区

    运行时常量池是方法区的一部分

    方法区是一个大数组,常量区是数组的一部分,类加载后的常量池表的内容存放到方法区的运行时常量池中

    6.直接内存

    不属于JVM的内存结构,是操作系统的内存,常见于NIO操作,主要用于数据缓冲区,它分配回收成本较高,但读写性能高

    NIO比常规的IO 快很多,

    因为常规IO读取文件的流程是: 磁盘文件 -》 系统缓存区 -》 java缓冲区。

    NIO的话是:磁盘文件 -》 直接内存。直接内存是java堆内存和系统内存都能访问的,java代码访问的时候就比上面的方式烧了一次复制的步骤,所以速度快

  • 相关阅读:
    Code For Better 谷歌开发者之声——初识Web与谷歌,拉起兴趣之心。
    SpringBoot中幕——配置文件properties与yml
    CV轻量级backbone模型小抄(1)
    css调整字体间距 以及让倾斜字体
    redis实现布隆过滤器
    spring boot flowable多人前加签
    Pariatur sint mollitia odit eveniet.Dazu Name tragen.
    ueditor百度富文本编辑器粘贴后html丢失class和style样式
    LeetCode_字符串反转_简单_557. 反转字符串中的单词 III
    Android使用glide时报错“ ����: �޷�����Fragment Glide.with(getContext()) ^ �Ҳ���and”
  • 原文地址:https://blog.csdn.net/KangYouWei6/article/details/133011642