欧美性猛交XXXX免费看蜜桃,成人网18免费韩国,亚洲国产成人精品区综合,欧美日韩一区二区三区高清不卡,亚洲综合一区二区精品久久

打開(kāi)APP
userphoto
未登錄

開(kāi)通VIP,暢享免費電子書(shū)等14項超值服

開(kāi)通VIP
Java中的JIT機制對運行速度的優(yōu)化
原文出處: lxydo

JIT的起源

在先前的博客,Javac編譯過(guò)程,簡(jiǎn)略講述了Java compiler(javac),可以看出javac和C的compiler不一樣, 并不是直接將 Java 的源代碼 編譯成成處理器的指令。 相反地,它產(chǎn)生的是統一規格、與機器 binary 格式無(wú)關(guān)的 bytecode。 在執行期,JVM 會(huì )逐條解釋執行 bytecode, 這是為甚么 Java 在跨平臺上會(huì )這么成功的主要原因, 你可以在某個(gè)平臺上寫(xiě)完、build 一份,然后在其他的平臺上頭執行。 但是這也導致了嚴重的問(wèn)題, interpret 通常比直接 compile 成 平臺限定的原生 binary 碼來(lái)得慢。 Sun 在 90 年代后期就已經(jīng)了解這個(gè)嚴重度, 當時(shí)他們請了 Cliff Click 博士來(lái)提供解決方案。

Hot Spot

他們在虛擬機中引入了JIT編譯器(即時(shí)編譯器),當虛擬機發(fā)現某個(gè)方法或代碼塊運行特別頻繁時(shí),就會(huì )把這些代碼認定為“Hot Spot Code”(熱點(diǎn)代碼),為了提高熱點(diǎn)代碼的執行效率,在運行時(shí),虛擬機將會(huì )把這些代碼編譯成與本地平臺相關(guān)的機器碼,并進(jìn)行各層次的優(yōu)化,完成這項任務(wù)的正是JIT編譯器。 在某些情況下,調整好的最佳化 JVM 效能可能超過(guò)手工的 C++ 或 C。 現在主流的商用虛擬機(如Sun HotSpot、IBM J9)中幾乎都同時(shí)包含解釋器和編譯器。當程序需要迅速啟動(dòng)和執行時(shí),解釋器可以首先發(fā)揮作用,省去編譯的時(shí)間,立即執行;當程序運行后,隨著(zhù)時(shí)間的推移,編譯器逐漸會(huì )返回作用,把越來(lái)越多的代碼編譯成本地代碼后,可以獲取更高的執行效率。解釋執行可以節約內存,而編譯執行可以提升效率。 運行過(guò)程中會(huì )被即時(shí)編譯器編譯的“Hot Spot Code”有兩類(lèi):

1
2
<code>· 被多次調用的方法。
· 被多次調用的循環(huán)體。 </code>

兩種情況,編譯器都是以整個(gè)方法作為編譯對象,這種編譯也是虛擬機中標準的編譯方式。要知道一段代碼或方法是不是熱點(diǎn)代碼,是不是需要觸發(fā)即時(shí)編譯,需要進(jìn)行Hot Spot Detection(熱點(diǎn)探測)。目前主要的熱點(diǎn) 判定方式有以下兩種:

  • 基于采樣的熱點(diǎn)探測:采用這種方法的虛擬機會(huì )周期性地檢查各個(gè)線(xiàn)程的棧頂,如果發(fā)現某些方法經(jīng)常出現在棧頂,那這段方法代碼就是“Hot Spot Code”。這種探測方法的好處是實(shí)現簡(jiǎn)單高效,還可以很容易地獲取方法調用關(guān)系,缺點(diǎn)是很難精確地確認一個(gè)方法的熱度,容易因為受到線(xiàn)程阻塞或別的外界因素的影響而擾亂熱點(diǎn)探測。
  • 基于計數器的熱點(diǎn)探測:采用這種方法的虛擬機會(huì )為每個(gè)方法,甚至是代碼塊建立計數器,統計方法的執行次數,如果執行次數超過(guò)一定的閥值,就認為它是“熱點(diǎn)方法”。這種統計方法實(shí)現復雜一些,需要為每個(gè)方法建立并維護計數器,而且不能直接獲取到方法的調用關(guān)系,但是它的統計結果相對更加精確嚴謹。

具體實(shí)例

下面看一段JIT提高程序性能的例子

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
class Calculator {
    Wrapper wrapper;
    public void calculate() {
        y = wrapper.get();
        z = wrapper.get();
        sum = y + z;
    }
}
class Wrapper {
    final int value;
    final int get() {
        return value;
    }
}

上面這是一段開(kāi)發(fā)人員寫(xiě)的代碼,假設這段代碼是 Hot Spot

1
2
3
4
5
6
7
8
9
10
11
12
13
14
class Calculator {
     Wrapper wrapper;
     public void calculate() {
         y = wrapper.value;
         sum = y + y;
     }
 }
 class Wrapper {
     final int value;
     final int get() {
         return value;
     }
 }

這是HotSpot VM經(jīng)過(guò)Hot Spot Detection 后對代碼進(jìn)行優(yōu)化的等價(jià)結果,當然JIT是將 bytecode 編譯成本地機器碼,這里展示的是優(yōu)化后與之等價(jià)的源代碼。 看上面的編譯優(yōu)化,首先是

  1. inline method(方法內聯(lián)) 用 b.value 取代 wrapper.get(), 不透過(guò)函數呼叫而直接存取 wrapper.value 來(lái)減少延遲。
1
2
3
4
5
6
7
8
class Calculator {
    Wrapper wrapper;
    public void calculate() {
        y = wrapper.value;
        z = wrapper.value;
        sum = y + z;
    }
}

2.移除多余的載入:用 z = y 取代 z = wrapper.value, 所以只存取區域變量而不是 wrapper.value 來(lái)減少延遲。

1
2
3
4
5
6
7
8
class Calculator {
    Wrapper wrapper;
    public void calculate() {
        y = wrapper.value;
        z = y;
        sum = y + z;
    }
}

3.copy propagation(復寫(xiě)傳播):用 y = y 取代 z = y, 沒(méi)有必要再用一個(gè)變量 z,因為 z 跟 y 會(huì )是相等的。

1
2
3
4
5
6
7
8
class Calculator {
    Wrapper wrapper;
    public void calculate() {
        y = wrapper.value;
        y = y;
        sum = y + y;
    }
}

4消除不用的源代碼:y = y 是不必要的,可以消滅掉。

1
2
3
4
5
6
7
class Calculator {
     Wrapper wrapper;
     public void calculate() {
         y = wrapper.value;
         sum = y + y;
     }
 }

以上是單個(gè)類(lèi)里面的優(yōu)化,下面看有繼承關(guān)系的優(yōu)化

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public interface Animal {
    public void eat();
}
public class Cat implements Animal{
    public void eat() {
        System.out.println("cat eat fish");
    }
}
public class Test{
    public void methodA(Animal animal){
        animal.eat();
    }
}

首先分析Animal的整個(gè)”類(lèi)型繼承關(guān)系”,發(fā)現只有一個(gè)實(shí)現類(lèi)Cat,那么在methodA(Animal animal)的代碼就可以?xún)?yōu)化為如下,

1
2
3
public void methodA(Animal animal){
       System.out.println("cat eat fish");
   }

但是,如果之后在運行過(guò)程中,”類(lèi)型繼承關(guān)系”發(fā)現Animal又多了一個(gè)實(shí)現類(lèi)Dog,那么此時(shí)就不在執行之前優(yōu)化編譯好的機器碼了,而是進(jìn)行解釋執行,即如下的”逆優(yōu)化”。 逆優(yōu)化: 當編譯后的機器碼的執行不再符合優(yōu)化條件,則該機器碼對應的部分回到解釋執行。 以上介紹的都是C1優(yōu)化,還有主要用于服務(wù)端程序優(yōu)化的C2優(yōu)化,這里就不再介紹了。

本站僅提供存儲服務(wù),所有內容均由用戶(hù)發(fā)布,如發(fā)現有害或侵權內容,請點(diǎn)擊舉報。
打開(kāi)APP,閱讀全文并永久保存 查看更多類(lèi)似文章
猜你喜歡
類(lèi)似文章
(3)Java內存對象的逃逸分析
【深入Java虛擬機】之七:Javac編譯與JIT編譯
為什么匿名內部類(lèi)調用的方法內局部變量必須為final
Python代碼運行慢怎么辦?這五種方法要謹記!
C#中的泛型
C# 泛型簡(jiǎn)介
更多類(lèi)似文章 >>
生活服務(wù)
分享 收藏 導長(cháng)圖 關(guān)注 下載文章
綁定賬號成功
后續可登錄賬號暢享VIP特權!
如果VIP功能使用有故障,
可點(diǎn)擊這里聯(lián)系客服!

聯(lián)系客服

欧美性猛交XXXX免费看蜜桃,成人网18免费韩国,亚洲国产成人精品区综合,欧美日韩一区二区三区高清不卡,亚洲综合一区二区精品久久