JVM连杀

  1. 说说JVM的内存区域,各区域的作用

    1. Java虚拟机栈
      线程私有的,即平时所说的栈内存,描述的是Java方法执行的内存模型。每个方法从调用到执行完成都对应着一个入栈和出栈的过程。其中栈帧用于存储局部变量表(存放基本数据类型,对象引用等)、操作数栈、动态链接、方法出口信息等。
      可能会有StackOverFlow错误和OutOfMemoryError错误

    2. 本地方法栈
      线程私有的,同上,只不过对应的是调用的系统本地方法(Native Method)

    3. 程序计数器
      线程私有的,当前线程执行的字节码的行号指示器。Java虚拟机规范中唯一一个没有规定任何OutofMemoryError情况的区域。

    4. Java堆
      线程共享的,垃圾回收关注的区域。用于存放对象实例,几乎所有的对象实例都是在这分配内存。

    5. 方法区
      线程共享的,存储已被虚拟机加载的类信息、常量、静态变量等。
      运行时常量池就在此区域。

  2. 如果创建一个数组,JVM如何分配内存
    在堆中分配数组空间,数组内存的是实际对象的指针,数组内的对象是需要另外在堆中分配空间的。

  3. 什么时候进行垃圾回收

    1. 在新生代的Eden区满了,会触发新生代GC(Minor GC),经过多次触发新生代GC存活下来的对象就会升级到老年代,升级到老年代的对象所需的内存大于老年代剩余的内存,则会触发老年代GC(Full GC)
    2. 主动调用System.gc()的时候也可能会触发Full GC
  4. JVM垃圾回收的算法
    复制算法: 快速,但浪费内存
    标记-清除算法: 快速,但会产生空间碎片,不利于分配大对象
    标记-整理算法: 速度不及前两个快,但是无空间碎片
    分代收集算法: 将堆内存分为不同的区域,对不同的区域使用不同的垃圾回收算法。
    那么,分代是如何划分的? 如下:
    一般根据对象的存货周期不同将java堆分为新生代(Eden,From Survivor,To Survivor)和年老带。新生代中的对象大多存活期较短,每次能回收很多而存活的少,所以在新生代使用复制算法;年老代中的对象存活率高,则使用标记-整理算法或者标记-清除算法。
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-NUg1ZfIs-1574127278489)(https://images2015.cnblogs.com/blog/345531/201511/345531-20151115204320728-1210139023.png)]
    那么,为啥要将新生代分为Eden和Survivor呢,为啥还要两个Survivor呢。
    分为Eden和Survivor这是因为复制算法需要移动现有对象,需要将内存分为两个区域(这两个区域的占比和通过-XX:SurvivorRatio=8来配置,默认是8:1:1);将Survivor分成两个是为了保证,留给新对象分配空间的区域永远都是最大的Eden区。

  5. JVM的垃圾回收器有哪些,特性和原理是啥

    1. 串行垃圾回收器(Serial/Serial Old)
      使用单线程处理垃圾回收,实现容易效率高,但无法利用多处理器的优势,适合单处理机和小内存的的机器应用。
    2. 并行垃圾回收器(ParNew/Parallel Scavenge(适合注重吞吐量的)/Parallel Old)
      多线程进行垃圾回收
    3. 并发垃圾回收器(CMS)
      GC线程和应用线程并发执行(但是初始标记和二次标记时仍然需要Stop-the-World),大大缩短停顿时间,适合响应时间优先的应用
    4. G1回收器
      待补充
  6. 如何判断对象空间能否被回收
    JVM中使用的是可达性分析法来判断对象是否是存活的。其基本思想就是,通过一系列"GC Roots"对象作为起始点,从这些节点开始向下搜索,形成一条引用链;当一个对象到任何"GC Roots"都没有任何一条引用链的时候(即不可达),那么该对象就是不可用的,(且没有在finalize()方法中被引用)就可以被回收。那么哪些对象能作为"GC Roots"呢,有如下几种:

    1. 虚拟机栈中引用的对象
    2. 方法区中类静态属性引用的对象
    3. 方法区中常量引用的对象
    4. 本地方法栈中JNI引用的对象

    总结起来其实就是,当前程序运行中使用到的对象~

  7. 如何查看JVM的GC情况
    jstat -gc pid

    S0C:第一个幸存区的大小
    S1C:第二个幸存区的大小
    S0U:第一个幸存区的使用大小
    S1U:第二个幸存区的使用大小
    EC:伊甸园区的大小
    EU:伊甸园区的使用大小
    OC:老年代大小
    OU:老年代使用大小
    MC:方法区大小
    MU:方法区使用大小
    CCSC:压缩类空间大小
    CCSU:压缩类空间使用大小
    YGC:年轻代垃圾回收次数
    YGCT:年轻代垃圾回收消耗时间
    FGC:老年代垃圾回收次数
    FGCT:老年代垃圾回收消耗时间
    GCT:垃圾回收消耗总时间
    
  8. 虚拟机类加载机制

    1. 何时加载
      从java编写层面理解的话:

      1. 一般就是new实例化对象的时候;
      2. 使用了一个类的静态字段的时候;
      3. 调用类的静态方法的时候;
      4. 使用反射相关调用的时候;
      5. 初始化类的时候如果其父类没有初始化的时候会先触发父类的初始化;
    2. 加载流程
      加载–>验证–>准备–>解析–>初始化–>使用–>卸载

    3. 双亲委派模式
      如果一个类加载器收到了类加载的请求,它会把这个请求委托给父类加载器去完成,只有当父类加载器反馈说无法加载这个类时,才会自己去加载这个类。
      当前JAVA加载器结构:
      Bootstrap Classloader <-- Extension Classloader <-- Application Classloader <-- User Classloader
      Bootstrap Classloader(启动类加载器):C++实现,在java里无法获取,负责加载/lib下指定的类
      Extension Classloader(扩展类加载器):java实现,负责加载lib/ext目录下的类
      Application Classloader(应用程序类加载器):用于加载classpath上指定的类库

      双亲委派模式的破坏:

      1. JNDI的实现
      2. OSGI的实现
        参考: 浅谈双亲委派和破坏双亲委派
  9. JMM模型
    定义了程序中各个变量的访问规则,其规定所有的变量(此变量非java中说的变量,指的是非线程私有的,eg 类属性对象,静态字段等)都存储在主内存中,每条线程还有自己的工作内存线程的工作内存中保存了该线程使用到的变量到主内存副本拷贝,线程对变量的所有操作(读取、赋值)都必须在工作内存中进行,而不能直接读写主内存中的变量。不同线程之间无法直接访问对方工作内存中的变量,线程间变量值的传递均需要在主内存来完成,对应定义了八个操作:

    lock: 作用于主内存的变量,把一个变量标识为一条线程独占状态。
    unlock: 作用于主内存变量,把一个处于锁定状态的变量释放出来,释放后的变量才可以被其他线程锁定。
    read: 作用于主内存变量,把一个变量值从主内存传输到线程的工作内存中,以便随后的load动作使用
    load: 作用于工作内存的变量,它把read操作从主内存中得到的变量值放入工作内存的变量副本中。
    use: 作用于工作内存的变量,把工作内存中的一个变量值传递给执行引擎,每当虚拟机遇到一个需要使用变量的值的字节码指令时将会执行这个操作。
    assign: 作用于工作内存的变量,它把一个从执行引擎接收到的值赋值给工作内存的变量,每当虚拟机遇到一个给变量赋值的字节码指令时执行这个操作。
    store: 作用于工作内存的变量,把工作内存中的一个变量的值传送到主内存中,以便随后的write的操作。
    write: 作用于主内存的变量,它把store操作从工作内存中一个变量的值传送到主内存的变量中。

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-dpMX14AQ-1574127278489)()]

    并定义了各操作的使用顺序和对应的约束规范,此处不细说,简化成如下happens-before规则:
    如果两个操作之间的关系不在此列,并且无法从下列规则推导出来的话,它们就没有顺序性保障,虚拟机可以对它们进行随意地重排序。

    程序次序规则:在一个线程内,按照程序代码顺序,书写在前面的操作先行发生于书写在后面的操作。准确地说应该是控制流顺序而不是程序代码顺序,因为要考虑分支、循环结构。
    管程锁定规则:一个unlock操作先行发生于后面对同一个锁的lock操作。这里必须强调的是同一个锁,而”后面“是指时间上的先后顺序。
    volatile变量规则:对一个volatile变量的写操作先行发生于后面对这个变量的读取操作,这里的”后面“同样指时间上的先后顺序。
    线程启动规则:Thread对象的start()方法先行发生于此线程的每一个动作。
    线终止规则:线程中的所有操作都先行发生于对此线程的终止检测,我们可以通过Thread.join()方法结束,Thread.isAlive()的返回值等作段检测到线程已经终止执行。
    线程中断规则:对线程interrupt()方法的调用先行发生于被中断线程的代码检测到中断事件的发生,可以通过Thread.interrupted()方法检测是否有中断发生。
    对象终结规则:一个对象初始化完成(构造方法执行完成)先行发生于它的finalize()方法的开始。
    传递性:如果操作A先行发生于操作B,操作B先行发生于操作C,那就可以得出操作A先行发生于操作C的结论。

    参考
    jmm
    《深入理解Java虚拟机 JVM高级特性与最佳实践》

//todo: 内存分配与回收策略

未完待续

发布了37 篇原创文章 · 获赞 58 · 访问量 16万+
展开阅读全文

没有更多推荐了,返回首页

©️2019 CSDN 皮肤主题: 大白 设计师: CSDN官方博客

分享到微信朋友圈

×

扫一扫,手机浏览