JVM系列之:JIT中的Virtual Call接口
文章目錄
- 簡介
- 最常用的接口List
- 多個(gè)List的調(diào)用
- 不一樣的List調(diào)用
- 總結(jié)
簡介
上一篇文章我們講解了Virtual Call的定義并舉例分析了Virtual Call在父類和子類中的優(yōu)化。
JIT對類可以進(jìn)行優(yōu)化,那么對于interface可不可以做同樣的優(yōu)化么?
一起來看看吧。
最常用的接口List
List應(yīng)該是大家最最常用的接口了,我想這個(gè)大家應(yīng)該不會反駁。
public interface List<E> extends Collection<E> {今天我們就拿List來做例子,體驗(yàn)一下JIT優(yōu)化接口的奧秘。
還是上代碼,要分析的代碼如下:
public class TestVirtualListCall {public static void main(String[] args) throws InterruptedException {List<String> list=new ArrayList<>();for (int i = 0; i < 10000; i++){doWithVMethod(list);}Thread.sleep(1000);}public static void doWithVMethod(List<String> list){list.add("www.flydean.com");} }如果在命令行運(yùn)行,大家記得在運(yùn)行時(shí)添加參數(shù)-XX:+UnlockDiagnosticVMOptions -XX:+PrintAssembly -XX:-Inline
直接看JIT Watcher的結(jié)果:
我們可以看到JIT中先對ArrayList的實(shí)現(xiàn)類做了一個(gè)比較。
然后調(diào)用的是invokeinterface,但是其本質(zhì)還是invokevirtual,并且我們可以看到這個(gè)調(diào)用是被優(yōu)化過了:optimized virtual call。
多個(gè)List的調(diào)用
同樣的,我們可以測試一下多個(gè)list子類的情況下怎么調(diào)用:
public class TestVirtualListCall2 {public static void main(String[] args) throws InterruptedException {List<String>[] lists=new List[]{new ArrayList<>(),new LinkedList<>()};for (int i = 0; i < 10000; i++){doWithVMethod(lists[i%2]);}Thread.sleep(1000);}public static void doWithVMethod(List<String> list){list.add("www.flydean.com");} }同樣,使用JIT Watcher來運(yùn)行:
我們可以看到JIT做了兩次對象類型的比較,然后對兩個(gè)invokeinterface都做了優(yōu)化。
結(jié)果和我們的父類子類結(jié)果是一樣的。
不一樣的List調(diào)用
上面我們在做多個(gè)list調(diào)用的時(shí)候,是輪循著來調(diào)用的,如果我們先調(diào)用ArrayList的方法,再調(diào)用LinkedList的方法,會有什么不同呢?
一起來看看。
public class TestVirtualListCall3 {public static void main(String[] args) throws InterruptedException {List<String> list1 = new ArrayList<>();List<String> list2 = new LinkedList<>();for (int i = 0; i < 10000; i++){doWithVMethod(list1);}Thread.sleep(1000);for (int i = 0; i < 10000; i++){doWithVMethod(list2);}Thread.sleep(1000);}public static void doWithVMethod(List<String> list){list.add("www.flydean.com");} }上面我們先循環(huán)ArrayList,然后再循環(huán)LinkedList。
看下結(jié)果有什么不同:
可以看到,JIT先比較了ArrayList,然后只做了一次方法的優(yōu)化。
也就是說LinkedList的調(diào)用是沒有進(jìn)行代碼優(yōu)化的。
上面的結(jié)果是在C2編譯器下,也就是level4的編譯水平下解析的。
我們看下如果在C1編譯器下,也就是Level3編譯水平下有什么不同。
可以看到C1編譯下,所有的invokeinterface都沒有進(jìn)行編譯優(yōu)化,只有在C2編譯下,才會進(jìn)行優(yōu)化。
不同的JVM版本可能優(yōu)化方式不一樣。大家可以自行實(shí)驗(yàn)。
總結(jié)
本文用實(shí)例展示了Virtual Call在interface上面的優(yōu)化使用。
感興趣的朋友,可以一起討論。
本文作者:flydean程序那些事
本文鏈接:http://www.flydean.com/jvm-virtual-call-interface/
本文來源:flydean的博客
歡迎關(guān)注我的公眾號:程序那些事,更多精彩等著您!
總結(jié)
以上是生活随笔為你收集整理的JVM系列之:JIT中的Virtual Call接口的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: JVM系列之:JIT中的Virtual
- 下一篇: 看动画学算法之:hashtable