JVM简介

JVM是Java Virtual Machine(Java虚拟机)的缩写,是实现java跨平台特性的关键。Java虚拟机在执行字节码时,把字节码解释成具体平台上的机器指令进行执行,这样实现了Java“一次编译,到处运行”。

JVM新生代

  1. 新生代=1个eden区+2个Survivor区
  2. -Xmn 年轻代大小(早期版本)
    -XX:NewSize, -XX: MaxNewSize (设置年轻代大小(for 1.3/1.4)默认值大小调整为个堆的3/8)
  3. -XX: NewRatio
    年轻代(包含Eden和两个Survivor区)与年老代的比值(除去持久代)Xms=Xmx并且设置了Xmn的情况下,改参数不需要设置。
  4. -XX.SurvivorRatio
    Eden区与Survivor区的大小比值,设置为8,则两个Survivor区与一个Eden区的比值为2:8,一个Survivor区占整个年轻代的1/10
  5. 用来存放JVM刚分配的Java对象。

Java年老代(tenured generation)

  1. 老年代=整个堆 - 年轻代大小 - 持久代大小
  • 年轻代中经过垃圾回收没有回收掉的对象被复制到年老代。
  • 老年代存储对象比年轻代年龄大的多,而且不乏大对象。
  • 新建的对象也有可能直接进入老年代
    • 大对象,可通过启动参数设置-XX: PretenureSizeThreshold=1024 (单位为字节,默认为0)来代表超过多大时就不在新生代分配,而是直接在老年代分配。
  • 大的数组对象,且数组中午引用外部对象。
  • 老年代大小无配置参数

Java持久代(perm generation)

  1. 持久代=整个堆 - 年轻代大小 - 老年代大小
  2. -XX:PermSize -XX:MaxPermSize
    设置持久代的大小,一般情况推荐把-XX:PermSize设置成XX:MaxPermSize的相同值,因为永久带大小的调整会导致堆内存需要触发fgc.
  3. 存放Class,Method元信息,其大小与项目的规模,类,方法的数量有关。一般设置为128M就足够,设置原则是预留30%的空间。
  • 永久代的回收方式
    • 常量池中的常量,无用的类信息,常量的回收很简单,没有引用了就会被回收。
    • 对于无用的类进行回收,必须保证三点:
      • 类的所有实例都已经被回收
      • 加载类的ClassLoader已经被回收
      • 类对象的Class对象没有被引用(即没有通过反射引用该类的地方)

JVM内存垃圾回收

JVM垃圾收集算法
  1. 引用计数算法
    每个对象有一个引用计数属性,新增一个引用时计数加1,引用释放时计数减1,计数为0时可以回收。此方法简单,无法解决对象互斥循环引用的问题,还有一个问题是如何解决精确计数。此种方式已经不再使用。
  2. 根搜索算法
    从GC Roots开始向下检索,搜索所走过的路径为引用链,当一个对象到GC链没有一个引用链相连时,则证明对象是不可用的,不可达对象。
    在Java语言中,GC Roots包括:
    • 虚拟机栈中引用的对象
    • 方法区中类静态属性实体引用的对象。
    • 方法区中常量引用的对象
    • 本地方法栈中JNI引用的对象
JVM垃圾回收算法
  1. 复制算法(Copying)
    复制算法采用从根集合扫描,并将存活的对象复制到一块新的,没有使用过空间中,这种算法当控件存活的对象比较少时,极为高效,但是此算法用于新生代内存回收,从E区回收到S0或者S1。
  • 标记清除算法
    标记-清除算法采用从根集合进行扫描,对存活的对象标记,标记完毕后,再扫描整个空间中未被标记的对象,进行回收。
    标记清除算法不需要进行对象的移动,并且仅对不存货的对象进行处理,在存活对象比较多的情况下极为高效,但是由于标记清除算法直接回收不存活的对象,因此会造成内存碎片。

  • 标记整理算法
    标记整理算法采用标记-清除算法一样的方式进行对象的标记,但在清除时不同,在回收不存活的对象占用空间后,会将所有存活对象往左端空闲空间移动,并更新对应的指针。标记整理算法是在标记清除算法的基础上,又进行了对象的移动,因此成本更高,但是却解决了内存碎片的问题。

  • 回收的方式

    • 串行回收:gc单线程内存回收,会暂停所有用户线程
    • 并行回收:收集是指多个gc线程并行工作,但此时用户线程是暂停的,所以 Serial是串行的,Parallel收集器是并行的,而CMS收集器是并发的。
    • 并发回收:是指用户线程同事执行(不一定是并行,可能交替,但总体上是在同时执行),不需要停顿用户线程——其实在CMS中用户线程还是需要停顿的,只是非常短,gc线程在另一个CPU上执行。

JVM的优化

JVM的调优,尽量的避免FULL GC。

  1. JVM堆配置参数

    • -Xms初始堆大小
      默认物理内存的1/64(<1GB)
    • -Xmx最大堆大小
      默认物理内存的1/4(<1GB),实际建议不大于4GB
    • 一般建议设置 -Xmx = -Xms
      好处是避免每次 在gc之后,调整堆的大小,减少系统内存分配开销。
    • 整个堆的大小=年轻代大小+年老代大小+持久代大小
  2. GC性能指标

    • 吞吐量 应用花在非GC上的时间百分比
    • GC负荷: 与吞吐量相反,指应用花在GC上的时间百分比
    • 暂停时间: 应用花在GC stop-the-world的时间
    • GC频率: GC频率
    • 反应速度:从一个对象成为垃圾到被回收的时间
    • 一个交互式的应用要求暂停时间越少越好,然而,一个非交互型的应用需要GC负荷越低越好
    • 一个实时系统对暂停时间和GC负荷的要求,越低越好。
  3. 内存容量配置原则

    • 年轻代大小选择

      • 响应时间优先的应用:尽可能设大,直到接近系统的最低响应时间限制,(根据实际情况选择),在此种情况下,年轻代收集发生的频率是最小的,同时,减少到达老年代的对象。

      • 吞吐量优先的应用: 尽可能的设置大,可能到达的Gbit的程度。因为对响应时间没有要求,垃圾收集可以并行进行,一般适合8CPU以上的应用。

    • 避免设置过小,当新生代设置过小会导致:

      • YGC次数更加频繁
      • 可能导致YGC对象直接进入旧生代,如果此时旧生代满了,会触发FGC
  • 年老代大小选择
    • 响应时间优先的应用: 年老代使用并发收集器CMS,所以大小需要小心设置,一般考虑并发会话率和会话持续时间等一些参数。如果堆设置小了,可能会造成内存碎片;高回收频率和应用暂停使用传统标记清除的方式;如果堆大了,则需要较长的收集时间。优化方案需要参考以下数据(通过GC log):

      • 并发垃圾收集信息
      • 持久代并发收集次数
      • 传统gc信息
      • 年轻代和年老代回收上的时间比例
    • 吞吐量优先的应用:一般吞吐量优先的应用都有一个很大的年轻代和一个较小的年老代,这样可以尽可能回收掉大部分短期对象,减少中期的对象,而年老代尽量存放长期存活的对象。

  • CMS(并发标记清除)回收器相关参数
    • 标记清除算法
      同时是一个使用多线程并发回收的垃圾收集器
    • -XX:ParallelCMSThreads
      手工设定CMS的线程数量,CMS默认启动的线程数是(ParallelGCThreads+3)/4
  • -XX:+UseConcMarkSweepGC开启
    使用ParNew+CMS+Serial Old的收集器组合进行内存回收,Serial Old作为CMS出现“Concurrent Mode Failure”失败后的后备收集器。
  • -XX:CMSInitiatingOccupancyFraction
    设置CMS收集器在年老代空间被使用多少后触发垃圾收集,默认值是68%,仅在CMS收集器时有效,-XX:CMSInitiatingOccupancyFraction=70
  • -XX:+UseCMSCompactAtFullCollection
    由于CMS收集器会产生碎片,此参数设置在垃圾收集后是否需要一次内存碎片整理过程,仅在使用CMS收集器时有效
  • -XX:+CMSFullGCBeforeCompaction
    设置CMS收集器在进行若干次垃圾收集后再次进行内存碎片整理过程,通常与UseCMSCompactAtFullCollection参数一起使用
  • -XX:CMSInitiatingPermOccupancyFraction
    设置Perm Gen使用到达多少比率时触发,默认92%

http://blog.csdn.net/ning109314/article/details/10411495/#comments

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

推荐阅读更多精彩内容

  • JVM架构 当一个程序启动之前,它的class会被类装载器装入方法区(Permanent区),执行引擎读取方法区的...
    cocohaifang阅读 1,650评论 0 7
  • Java 虚拟机有自己完善的硬件架构, 如处理器、堆栈、寄存器等,还具有相应的指令系统。JVM 屏蔽了与具体操作系...
    尹小凯阅读 1,685评论 0 10
  • 原文阅读 前言 这段时间懈怠了,罪过! 最近看到有同事也开始用上了微信公众号写博客了,挺好的~给他们点赞,这博客我...
    码农戏码阅读 5,954评论 2 31
  • 1.一些概念 1.1.数据类型 Java虚拟机中,数据类型可以分为两类:基本类型和引用类型。基本类型的变量保存原始...
    落落落落大大方方阅读 4,524评论 4 86
  • 这篇文章是我之前翻阅了不少的书籍以及从网络上收集的一些资料的整理,因此不免有一些不准确的地方,同时不同JDK版本的...
    高广超阅读 15,572评论 3 83