美文网首页
2021-02-23-jvm

2021-02-23-jvm

作者: 你家门口的两朵云 | 来源:发表于2021-02-23 09:02 被阅读0次
    1、请介绍⼀下 JVM 内存模型?⽤过什么垃圾回收器

    Java内存模型结构分为线程共享区线程私有区

    线程共享区:堆、方法区

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

    堆:用于存放对象实例和数组,由于堆是用来存放对象实例,因此堆也是垃圾收集器管理的主要区域,故也称为GC堆。由于现在的垃圾收集器基本都采用分代收集算法,所以堆的内部结构只包含新生代和老年代。

    方法区:用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
         方法区通常和永久区(Perm)关联在一起,但永久代与方法区不是一个概念,只是有的虚拟机用永久代来实现方法区,这样就可以用永久代GC来管理方法区,省去专门内存管理的工作
         根据Java虚拟机规范的规定,当方法区无法满足内存分配的需求时,将抛出 OutOfMemoryError 异常
    虚拟机栈
         每个方法在执行的时候都会创建一个栈帧,用于存储局部变量表、操作数栈、动态链接、方法出口等信息
         每个方法从调用直至完成的过程,对应一个栈帧在虚拟机栈中入栈到出栈的过程
         局部变量表主要存放一些基本类型的变量和对象句柄,它们可以是方法参数,也可以是方法的局部变量
    程序计数器
       为什么需要程序计数器?
         在多线程情况下,当线程数超过CPU数量或CPU内核数量时,线程之间就要根据时间片轮询抢夺CPU时间资源。也就是说,在任何一个确定的时刻,一个处理器都只会执行一条线程中的指令。


    2、出现 GC 的问题,你是如何定位、解决问题的,说说解决思路和处理⽅法?

    程序计数器、虚拟机栈、本地方法栈、堆区、方法区。其中程序计数器、虚拟机栈、本地方法栈3个区域随线程而生、随线程而灭,因此这几个区域的内存分配和回收都具备确定性,就不需要过多考虑回收的问题,因为方法结束或者线程结束时,内存自然就跟随着回收了。而Java堆区和方法区则与之不同,这部分内存的分配和回收是动态的,正是垃圾收集器所需关注的部分。
      垃圾定位分析:
        有两种方式,一种是引用计数(但是无法解决循环引用的问题);另一种就是可达性分析

    引用计数
    JAVA中当一个对象被创建的时候会给该对象分配一个变量,这个变量便称为对象的引用。当任何其它变量被赋值为这个对象的引用时,计数加1。但当一个对象实例的某个引用超过了生命周期或者被设置为一个新值时,对象实例的引用计数器减1。任何引用计数器为0的对象实例可以被当作垃圾收集。 此种处理方式是最快速的。但是有bug,相互引用的变量永远无法为0

    可达性分析
    可达性分析算法是从离散数学中的图论引入的,程序把所有的引用关系看作一张图,从一个节点GC ROOT开始,寻找对应的引用节点,找到这个节点以后,继续寻找这个节点的引用节点,当所有的引用节点寻找完毕之后,剩余的节点则被认为是没有被引用到的节点,即无用的节点,无用的节点将会被判定为是可回收的对象。

    在Java语言中,可作为GC Roots的对象包括下面几种:
      a) 虚拟机栈中引用的对象(栈帧中的本地变量表);
      b) 方法区中类静态属性引用的对象;
      c) 方法区中常量引用的对象;
      d) 本地方法栈中JNI(Native方法)引用的对象

    相关文章

      网友评论

          本文标题:2021-02-23-jvm

          本文链接:https://www.haomeiwen.com/subject/hlalfltx.html