• java虚拟机——轻松搞懂jvm


    一、JVM体系结构概述

    • JVM位置

    • JVM体系结构

    1.1 类加载器 ClassLoader

      类加载器(ClassLoader)负责加载class文件,class文件在文件开头有特定的文件标示,并且ClassLoader只负责class文件的加载,至于它是否可以运行,则由Execution Engine决定。

    大致流程图如下图所示:

    1.1.1 类加载器的分类

      在java中类加载器共分为两大类,分别是:虚拟机自带的加载器用户自定义加载器
    其中:

    虚拟机自带的加载器

    • 启动类加载器(Bootstrap)C++

    • 扩展类加载器(Extension)Java

    • 应用程序类加载器(AppClassLoader)Java(也叫系统类加载器,加载当前应用的classpath的所有类)

    用户自定义加载器

    • Java.lang.ClassLoader的子类,用户可以定制类的加载方式

    1.1.2 类加载器的加载过程

      类加载器对于类的加载有两个机制,分别是:双亲委派机制和砂箱安全机制。

      类加载器加载类的过程如图所示:

      从图中我们可以看出,当一个Hello.class这样的文件要被加载时。不考虑我们自定义类加载器,首先会在AppClassLoader中检查是否加载过,如果有那就无需再加载了。如果没有,那么会拿到父加载器,然后调用父加载器的loadClass方法。父类中同理会先检查自己是否已经加载过,如果没有再往上。注意这个过程,直到到达Bootstrap classLoader之前,都是没有哪个加载器自己选择加载的。如果父加载器无法加载,会下沉到子加载器去加载,一直到最底层,如果没有任何加载器能加载,就会抛出ClassNotFoundException。

    1.2 本地方法栈

      至于jvm中的本地方法栈,该方法栈是针对native方法来进行压栈的。具体过程为:

      执行native方法的时候需要将native方法压入到本地方法栈,本地方法栈中的方法就需要操作系统来执行,这个时候需要执行引擎Execution Engine 来负责进行解释,然后调用执行本地方法接口(这时会用到本地方法库)。

    1.3 程序计数器

    程序计数器就是一个小小的指针(是一个非常小的内存空间,几乎可以忽略不记),通过它来确定执行完一个方法之后再执行下一个方法。其所处位置类似于两节车厢的连接处。

    1.4 Execution Engine

    Execution Engine执行引擎负责解释命令,提交操作系统执行。

    1.5 方法区 Method Area

      方法区是被所有线程共享,所有字段和方法字节码,以及一些特殊方法如构造函数,接口代码也在此定义。简单说,所有定义的方法的信息都保存在该区域,此区属于共享区间

    静态变量 + 常量 + 类信息(构造方法/接口定义) + 运行时常量池存在方法区中

      但是一定要注意:实例变量存在于堆内存中,和方法区无关。

    1.6 栈 Stack

    1.6.1 栈是什么

      栈也叫栈内存,主管Java程序的运行,是在线程创建时创建,它的生命期是跟随线程的生命期,线程结束栈内存也就释放,对于栈来说不存在垃圾回收问题,只要线程一结束该栈就Over,生命周期和线程一致,是线程私有的。8种基本类型的变量+对象的引用变量+实例方法都是在函数的栈内存中分配。

    1.6.2 栈运行原理

      栈中的数据都是以栈帧(Stack Frame)的格式存在,栈帧是一个内存区块,是一个数据集,是一个有关方法(Method)和运行期数据的数据集,当一个方法A被调用时就产生了一个栈帧 F1,并被压入到栈中,
      A方法又调用了 B方法,于是产生栈帧 F2 也被压入栈,
      B方法又调用了 C方法,于是产生栈帧 F3 也被压入栈,
      ……
      执行完毕后,先弹出F3栈帧,再弹出F2栈帧,再弹出F1栈帧……

      遵循“先进后出”/“后进先出”原则。

    1.7 堆(Heap)体系结构

      堆内具体划分得看从逻辑上划分还是从物理上划分。

    1.7.1 从逻辑上划分

      堆从逻辑上分为新生区、养老区和永久区
      其中新生区又分为伊甸园区幸存者区,幸存区分为幸存0区(from区)幸存1区(to区)
      大致结构图如下图所示:

      new的对象首先被放到伊甸园区,如果经过垃圾回收这个对象没有被回收掉,那么会把它移动到幸存0区,
      如果再次经过垃圾回收这个对象仍然没有被回收掉,那么这次会把它移动到幸存1区,再垃圾回收,仍然幸存的话,就移动到养老区。
      在养老区之后,就不会被minor GC垃圾回收。

    1.7.2 垃圾回收 GC

      垃圾回收分为minor GC 和full GC,其中minor GC针对 Eden区和from区,full GC针对新生区和养老区。

    垃圾回收过程:

    新new的对象在伊甸园区(Eden)出生,随着对象越来越多,如果要是放不下的话,会触发轻量级的垃圾回收(minor GC),minor GC会回收Eden区和from区。
      针对Eden区,minor GC垃圾回收后,会将Eden中的幸存者移到to区。
      针对from区,minor GC垃圾回收后,会根据from区中的对象的age来先进行一个判断,如果from中的对象的age<15那么会将该对象移动到to区,如果age>=15就会将该对象移动到养老区,这些对象的age,每移动一次,其age就会加1。
    这个时候,Eden区中的幸存对象移动到了to区,from区中age<15的也移动到了to区,然后把Eden区清空,from区也清空,
      然后from和to区交换位置,即from区变成了to区,to区变成了from区。因此:to总为空。

    如此不断循环。

    特殊情况:如果new的对象比较大,伊甸园区放不下的话,就会在养老区创建,如果养老区空间也不够的话,直接就报oom了。

    经过不断的minor GC,养老区中的对象会越来越多,如果养老区空间放不下的话,就会触发Full GC,Full GC对新生区和养老区都会进行回收,如果Full GC也回收不了内存的话,就会出现OOM。

    产生OOM有两个原因:

    (1)Java虚拟机的堆内存设置不够,可以通过参数-Xms、-Xmx来调整。
    (2)代码中创建了大量大对象,并且长时间不能被垃圾收集器收集(存在被引用)。

    1.7.3 永久区

      永久存储区是一个常驻内存区域,用于存放JDK自身所携带的 Class,Interface 的元数据,也就是说它存储的是运行环境必须的类信息,被装载进此区域的数据是不会被垃圾回收器回收掉的,关闭 JVM 才会释放此区域所占用的内存。

      如果出现java.lang.OutOfMemoryError: PermGen space,说明是Java虚拟机对永久代Perm内存设置不够。一般出现这种情况,都是程序启动需要加载大量的第三方jar包。例如:在一个Tomcat下部署了太多的应用。或者大量动态反射生成的类不断被加载,最终导致Perm区被占满。

    Jdk1.6及之前: 有永久代, 常量池1.6在方法区

    Jdk1.7: 有永久代,但已经逐步“去永久代”,常量池1.7在堆

    Jdk1.8及之后: 无永久代,常量池1.8在元空间

    其中1.1.2节参考博文原文链接:https://blog.csdn.net/codeyanbao/article/details/82875064

  • 相关阅读:
    丑数——剑指offer面试题34
    把整数排成最小的数——剑指offer面试题33
    从1到n整数中1出现的次数——剑指offer面试题32
    各种排序方法及其比较
    scrapy安装
    水仙花数
    分数化小数(decimal)
    子序列的和
    倒三角
    韩信点兵
  • 原文地址:https://www.cnblogs.com/zhqin/p/12218129.html
Copyright © 2020-2023  润新知