欢迎访问昆山宝鼎软件有限公司网站! 设为首页 | 网站地图 | XML | RSS订阅 | 宝鼎邮箱 | 后台管理


新闻资讯

MENU

软件开发知识
原文出处: 五月的仓颉

媒介

http://www.cnblogs.com/xrq730/p/4839245.html,HotSpot回收的是表明器+编译器并存的架构,之前的这篇文章内里已经讲过了,本文只是把即时编译器这块再讲得详细一点罢了。虽然,其实本文的内容也没多大意义,90%都是观念上的对象,对付实际开拓、实际办理项目内里的疑难问题并没有什么太大的辅佐,只要看过就好了。

编译工具与触发条件

之前讲过,Sun利用的虚拟机之所以被叫做”HotSpot”,就是因为运行进程中会检测热点代码,那么运行进程中,昆山软件开发,会被即时编译器编译的”热点代码”有两类,即:

  • 被多次挪用的要领
  • 被多次执行的轮回体
  • 前者很好领略,一个要领被挪用得多了,要领体内代码执行的次数自然就多,他成为”热点代码”也是理所虽然。尔后者则是为了办理一个要领只被挪用过一次可能少量的屡次,可是要领体内部存在轮回次数较多的轮回体问题,这样轮回体的代码也被反复执行多次,因此这些代码也应该认为是”热点代码”。

    那上面的问题描写中,所谓”多次”都不是一个详细、严谨的用语,那么几多次才算”多次”?尚有,虚拟机如何统计一个要领或一段代码被执行过几多次呢?

    判定一段代码是不是热点代码,是不是需要触发即时编译,这样的行为称为”热点探测“,其实热点探测并不必然要知道要领详细被挪用了几多次,今朝主要的热点探测鉴定方法有两种:

  • 基于采样的热点探测
  • 基于计数器的热点探测
  • HotSpot虚拟机中利用的是第二种基于计数器的热点探测要领,它为每个要领筹备了两类计数器:要领挪用计数器和回边计数器。在确定虚拟机运行参数的前提下,这两个计数器都有一个确定的阈值,当计数器高出阈值溢出了,就会触发JIT编译,别离看一下:

    1、要领挪用计数器

    顾名思义,这个计数器就是用于统计要领被挪用的次数,它的默认阈值在Client模式下是1500次,在Server模式下是10000次。这个阈值可以通过参数-XX:CompileThreshold来工钱设定。当一个要领被挪用时,会查抄要领是否存在被JIT编译过的版本,假如存在,则优先利用编译后的当地代码来执行。假如不存在已被编译过的版本,则将此要领的挪用计数器值加1,然后判定要领挪用计数器和回边计数器值之和是否高出要领挪用计数器的阈值。假如已经高出阈值,那么将会向即时编译器提交一个该要领的代码编译请求。

    假如这个参数不做任何配置,那么要领挪用计数器统计的并不是要领被挪用的绝对次数,而是一个相对的执行频率,即一段时间之内要领被挪用的次数。当高出必然的时间限度,假如要领的挪用次数仍然不敷以让它提交给即时编译器编译,那这个要领的挪用计数器就会少一半,这个进程称为要领的挪用计数器热度的衰减,而这段时间就称为此要领统计的半衰周期。举办热度衰减的行动实在虚拟机举办垃圾接纳时顺便举办的,可以利用虚拟机参数-XX:-UseCounterDecay来封锁热度衰减,让要领计数器统计要领挪用的绝对次数,这样,只要系统运行时间足够长,绝大部门要领城市被编译成当地代码。别的,可以利用-XX:CounterHalfLifeTime参数配置半衰周期的时间,单元是秒。

    那假如参数不配置的话,执行引擎并不会同步期待编译请求完成,而是直接进入表明器凭据表明要领执行字节码,直到提交的请求被编译器编译完成。当编译事情完成之后,这个要领的挪用进口地点就会被系统自动改写成新的,下一次挪用该要领时就会利用已编译的版本。

    2、回边计数器

    它的浸染是统计一个要领中轮回体代码执行的次数,在字节码中碰着节制流向后跳转的指令称为”回边”。显然,成立回边技能其统计的目标就是为了触发OSR编译。关于回边计数器的阈值,固然HotSpot也提供了一个雷同于要领挪用计数器阈值-XX:CompileThreshold的参数-XX:BackEdgeThreshold供用户配置,可是当前虚拟机实际上并未利用此参数,因此我们需要配置别的一个参数-XX:OnStackReplacePercentage来间接调解回边计数器的阈值,其计较公式如下:

    (1)Client模式

    要领挪用计数器阈值 × OSR比率 / 1000,个中OSR比率默认值933,假如都取默认值,Client模式下回边计数器的阈值应该是13995

    (2)Server模式

    要领挪用计数器阈值 × (OSR比率 – 表明器监控比率) / 100,个中OSR比率默认140,表明器监控比率默认33,假如都取默认值,Server模式下回边计数器阈值应该是10700