JVM(二)垃圾回收器和内存分配策略

  • 判断对象的存活

    1. 引用计数法
      1> 优点:快,方便,实现简单
      2> 缺点:相互引用存在问题
    2. 可达性分析
      1> 在java,可作为GC Roots的对象包括:
      a, 方法区:类静态属性引用的对象
      b, 方法区:常量引用的对象
      c, 虚拟机栈(本地变量表)中引用的对象
      d, 本地方法栈JNI(Native方法)中引用的对象
  • 各种引用

    1. 强引用
      1> Object o = new Object()
    2. 软引用(SoftReference)
      1> 一些有用但是并非必需,用软引用关联的对象,系统将要发生OOM之前这些对象就会被回收
      2> User u = new User(); SoftReference<User> SoftReference = new SoftReference(u); u = null; //OOM
      3> 用在内存资源紧张的情况下
    3. 弱引用(WeakReference)
      1> 一些有用(有用的程度比软引用更低)但是并非必需,用弱引用关联的对象,只能生存到下次垃圾回收之前,GC发生时,不管内存是否足够都会被回收
      2> User u = new User(); WeakReference<User> WeakReference = new WeakReference(u); u = null; System.gc()
      3> 用在内存资源紧张的情况下
      4> WeakHashMap就是使用了弱引用
    4. 虚引用(PhantomReference)
      1> 幽灵有用,最弱,被垃圾回收的时候收到一个通知
  • 垃圾回收算法

    1. 标记-清除算法(Mark-Sweep)
      1> 会产生碎片,造成资源浪费

    2. 复制算法(Coping)
      1> 浪费空间,总会有一半空间无法使用
      2> 会移动数据,耗时

    3. 标记-整理算法(Mark-Compact)
      1> 会移动数据,耗时

    4. java虚拟机把算法都使用上,分代收集
      1> 对象的生命周期不同,90%以上的对象朝生夕死
      2> 新生代:刚new的对象,采用复制算法,Minor GC

       a, Eden 1个
       b, Survivor 2个
       c, 空间分配比例8(Eden):1(Survivor0):1(Survivor1)
      

      3> 老年代:对象存活超过一定时间(GC次数)的对象,标记-整理或标记清除,Full GC。调优是希望尽可能少的FULL GC
      4> 新生代经过一定次数的垃圾回收后依然未被回收的对象,会被移动到老年代区域,默认次数15

  • 垃圾回收器列表

    1. Serial 新生代 复制算法 单线程

    2. ParNew 新生代 复制算法 并行的多线程

    3. Parallel Scavenge 新生代 复制算法 并行的多线程

    4. Serial Old 老年代 标记整理 单线程

    5. Parallel Old 老年代 标记整理 并行的多线程收集器

    6. CMS 老年代 标记清除 并行与并发收集器


      分代收集.png
    7. G1(jdk7) 跨老年代和新生代 标记整理 + 化整为零 并行与并发收集器

    8. 垃圾收集的并行和并发
      1> 并行:垃圾收集的多线程的同时进行
      2> 并发:垃圾收集的多线程和应用的多线程同时进行

    9. 查看垃圾回收器:java -XX:+PrintCommandLineFlags -version

    10. 特点
      1> Serial/Serial Old 最古老的,单线程,独占式,成熟,适用于单CPU

      a, -XX:UseSerailGC 新生代和老年代都用串行收集器
      b, -XX:UseParNewGC 新生代使用ParNew,老年代使用Serial Old 
      c, -XX:UseParallelGC 新生代使用ParallelGC,老年代使用Serial Old
      

    2> ParNew和Serial基本没区别,唯一区别:多线程,适用于多CPU,停顿时间比Serial少

      a, -XX:UseParNewGC 新生代使用ParNew,老年代使用Serial Old 
    
垃圾回收器工作示意图.png

3> Parallel Scavenge(ParallelGC)/Parallel Old 关注吞吐量

        a, 停顿时间短的适合用户交互的程序(桌面应用),提高用户体验。
        b, 关注吞吐量,适用于后台计算任务. 
        c, -XX:MaxGCPauseMills 控制最大停顿时间 
        d, -XX:GCTimeRatio 允许的GC时间占总时间的比率,吞吐量的导数,整数 允许的GC时间 比如19 1/1+19 = 5%
        e, -XX:+UseAdaptiveSizePolicy 自适应的调整策略
        d, -XX:+UsePrallerOldGC: 新生代ParallelGC,老年代 Parallel Old

4> CMS(Concurrent Mark Sweep) 关注最短停顿时间,标记清除

        a, CMS运行的过程
            (1) 初始标记:时间短暂,GC-Roots能直接关联的标记
            (2) 并发标记:和用户的应用程序同时进行
            (3) 重新标记:时间短暂
            (4) 并发清除:和用户的应用程序同时进行
        b, -XX:UseConcMarkSweepGC 表示老年代的用CMS,新生代使用ParNew
        c, 浮动垃圾:用户的线程还在运行,需要给用户线程留下运行内存空间。
        d, -XX:CMSInitialOccupyFraction当老年代的空间超过这个值时启动收集,默认68%(jdk6) -> 92%(jdk7以后)。
        e, 不够时,产生错误:Concurrent Mode Failure,启动Serial Old收集器
        f, -XX:UseCMSCompactAtFullCollection 需要Full GC的时候开启内存碎片的整理,无法并发,默认开启
        g, -XX:CMSFullGCsBeforeCompaction 设置多少次不压缩的FullGC后来一次要压缩的,默认为0
垃圾回收器工作示意图.png

5> G1(Garbage First)

        a, jdk7才正式引入,到jdk9默认是收集器。采用分区回收的思维,基本不牺牲吞吐量的前提下完成低停顿的内存回收;可以预测的提顿是其最大的优势
        b, 面向服务端应用的垃圾回收器,目标取代CMS
        c, 整体回收算法是标记整理,局部回收算法是复制算法
        d, 设置参数 -XX:+UseG1GC
        e, 并行和并发,分代收集,空间整合,没有空间碎片,可预测的停顿
        f, G1收集运行
            (1) 初始标记:时间短暂,GC-Roots能直接关联的标记,产生一个全局停顿,都会有一个新生代的GC
            (2) 根区域扫描:扫描Survivor区可以直接到达老年代的区域
            (3) 并发标记阶段
            (4) 重新标记
            (5) 独占清理
            (6) 并发清理
        g, G1收集的几个阶段
            (1)新生代GC
            (2)并发标记周期
            (3)混合收集
            (4)可能的FullGC
        h, -XX:MaxGCPauseMillis 指定目标的最大停顿时间,G1尝试调整新生代和老年代的比例,堆大小,晋升年龄
        i, -XX:ParallerGCThreads GC的工作线程数量
G1垃圾回收器工作示意图.png

G1收集的阶段.png

11. 吞吐量 = 运行用户代码时间/(运行用户代码时间 + 垃圾收集时间)
12. 垃圾收集时间 = 垃圾回收频率 * 单次垃圾回收时间
13. 未来的垃圾回收器,JDK11中的ZGC,一种可扩展的低延迟垃圾收集器

    1> 处理TB量级的堆
    2> GC时间不超过10ms
    3> 与使用G1相比,应用吞吐量的降低不超过15%
    4> 实现技术:有色指针和加载屏障
14. Stop The World现象:无法避免,调优尽量减少
  • 内存分配与回收策略

    1. 对象优先在Eden分配:如果内存Eden空间不足,就会发生Minor GC
    2. 大对象直接进入老年代:大对象:很长的字符串和大型数组。
      1> 导致内存有空间,提前垃圾回收来放他们
      2> 进行大量的内存复制,影响Minor GC。
      3> -XX:PretenureSizeThreshold 参数 :大于参数值直接在老年代分配,缺省为0 表示绝不会直接分配在老年代
    3. 长期存活的对象将进入老年代,默认15
      1> -XX:MaxTenuringThreshold 参数
    4. 动态对象年龄判定
    5. 空间分配担保
      1> 大量的对象存活,survival空间不够,只要老年代连续空间大于新生代对象的总大小或者历次晋升的平均大小,就进行MinorGC,否则Full GC
  • 内存泄漏和内存溢出辨析

    1. 内存溢出是对象数据太多,超过了给定的内存空间
    2. 内存泄漏是指大量无用的对象一直未被回收,导致给定的虚拟机内存不够使用。譬如集合等容器中的对象一直往里面添加数据,不使用的也不清理
  • JDK为我们提供的工具

    1. jps 虚拟机进程状况工具,列出当前机器上正在运行的虚拟机进程
    2. jstat 虚拟机统计信息监视工具
      1> 假设需要每250毫秒查询一次进程2764垃圾收集状况,一共查询20次,那命令是:jstat -gc 2764(jvm进程号) 250 20
      2> -class 类加载器
      3> -compiler JIT
      4> -gc GC堆状态
      5> -gccapacity 各区大小
      6> -gccause 最近一次GC统计和原因
      7> -gcnew 新区统计
      8> -gcnewcapacity 新区大小
      9> -gcold 老区统计
      10> -gcpermcapacity 永久区大小
      11> -gcutil GC统计汇总
      12> -printcompilation HotSpot编译统计
    3. jinfo java 配置信息工具
      1> 查看和修改虚拟机参数
      2> -flag 查看虚拟机参数标记,也可以修改标记
    4. jmap java内存映像工具
      1> 在windows下受限
    5. jhat 虚拟机堆转存储快照分析工具
    6. jstack java堆栈跟踪工具
    7. jConsole java监视与管理控制台
      1>管理远程进程
      a, -Djava.rmi.server.hostname=..
      b, -Dcom.sun.management.jmxremote
      c, -Dcom.sun.management.jmxremote.port=8888
      d, -Dcom.sun.management.jmxremote.authenticate=false
      e, -Dcom.sun.management.jmxremote.ssl=false
    8. VisualVM 多合一故障处理工具
    9. 通过代码获取栈信息Thread.getAllStackTrace()
  • 了解MAT:一种eclipse的独立插件用于分析内存使用的工具,可以分析是否有内存泄漏等

  • 浅堆和深堆
    1)浅堆 这个对象的结构本身所占有的内存大小
    2)深堆 这个对象被GC回收后,可以真实释放的内存大小

  • 可以使用ManagementFactory类获取虚拟机的信息

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351