一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務(wù)器之家 - 編程語言 - Java教程 - JVM系列之:JIT中的Virtual Call接口操作

JVM系列之:JIT中的Virtual Call接口操作

2020-09-15 01:14flydean程序那些事 Java教程

這篇文章主要介紹了JVM系列之:JIT中的Virtual Call接口操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧

簡介

上一篇文章我們講解了Virtual Call的定義并舉例分析了Virtual Call在父類和子類中的優(yōu)化。

JIT對類可以進(jìn)行優(yōu)化,那么對于interface可不可以做同樣的優(yōu)化么?

一起來看看吧。

最常用的接口List

List應(yīng)該是大家最最常用的接口了,我想這個大家應(yīng)該不會反駁。

public interface List<E> extends Collection<E> {

今天我們就拿List來做例子,體驗一下JIT優(yōu)化接口的奧秘。

還是上代碼,要分析的代碼如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
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("m.ythuaji.com.cn");
  }
}

如果在命令行運行,大家記得在運行時添加參數(shù)-XX:+UnlockDiagnosticVMOptions -XX:+PrintAssembly -XX:-Inline

直接看JIT Watcher的結(jié)果:

JVM系列之:JIT中的Virtual Call接口操作

我們可以看到JIT中先對ArrayList的實現(xiàn)類做了一個比較。

然后調(diào)用的是invokeinterface,但是其本質(zhì)還是invokevirtual,并且我們可以看到這個調(diào)用是被優(yōu)化過了:optimized virtual call。

多個List的調(diào)用

同樣的,我們可以測試一下多個list子類的情況下怎么調(diào)用:

JVM系列之:JIT中的Virtual Call接口操作

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
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("m.ythuaji.com.cn");
  }
}

同樣,使用JIT Watcher來運行:

我們可以看到JIT做了兩次對象類型的比較,然后對兩個invokeinterface都做了優(yōu)化。

結(jié)果和我們的父類子類結(jié)果是一樣的。

不一樣的List調(diào)用

上面我們在做多個list調(diào)用的時候,是輪循著來調(diào)用的,如果我們先調(diào)用ArrayList的方法,再調(diào)用LinkedList的方法,會有什么不同呢?

一起來看看。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
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("m.ythuaji.com.cn");
  }
}

上面我們先循環(huán)ArrayList,然后再循環(huán)LinkedList。

看下結(jié)果有什么不同:

JVM系列之:JIT中的Virtual Call接口操作

可以看到,JIT先比較了ArrayList,然后只做了一次方法的優(yōu)化。

也就是說LinkedList的調(diào)用是沒有進(jìn)行代碼優(yōu)化的。

上面的結(jié)果是在C2編譯器下,也就是level4的編譯水平下解析的。

我們看下如果在C1編譯器下,也就是Level3編譯水平下有什么不同。

JVM系列之:JIT中的Virtual Call接口操作

可以看到C1編譯下,所有的invokeinterface都沒有進(jìn)行編譯優(yōu)化,只有在C2編譯下,才會進(jìn)行優(yōu)化。

不同的JVM版本可能優(yōu)化方式不一樣。大家可以自行實驗。

總結(jié)

本文用實例展示了Virtual Call在interface上面的優(yōu)化使用。

感興趣的朋友,可以一起討論。

補充知識:Java 8 Stream 流已被操作或關(guān)閉

在Java 8中,Stream不能重復(fù)使用,一旦被消耗或使用,流將被關(guān)閉,類似流水線,水龍頭的水一樣一去不復(fù)返

示例 - 流關(guān)閉

查看以下示例,它會拋出一個IllegalStateException,表示“流被關(guān)閉”。

TestJava8.java

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
package com.mkyong.java8;
import java.util.Arrays;
import java.util.stream.Stream;
public class TestJava8 {
  public static void main(String[] args) {
    String[] array = {"a", "b", "c", "d", "e"};
    Stream<String> stream = Arrays.stream(array);
    // loop a stream
    stream.forEach(x -> System.out.println(x));
    // reuse it to filter again! throws IllegalStateException
    long count = stream.filter(x -> "b".equals(x)).count();
    System.out.println(count);
  }
}

Output

?
1
2
3
4
5
6
7
8
9
10
11
12
java.lang.IllegalStateException: stream has already been operated upon or closed
 at java.util.stream.AbstractPipeline.(AbstractPipeline.java:203)
 at java.util.stream.ReferencePipeline.(ReferencePipeline.java:94)
 at java.util.stream.ReferencePipeline$StatelessOp.(ReferencePipeline.java:618)
 at java.util.stream.ReferencePipeline$2.(ReferencePipeline.java:163)
 at java.util.stream.ReferencePipeline.filter(ReferencePipeline.java:162)
 at com.hostingcompass.whois.range.run.TestJava8.main(TestJava8.java:25)
 at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
 at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
 at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
 at java.lang.reflect.Method.invoke(Method.java:498)
 at com.intellij.rt.execution.application.AppMain.main(AppMain.java:144)

示例 - 重用流

TestJava8.java

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
package com.mkyong.java8;
import java.util.function.Supplier;
import java.util.stream.Stream;
 public class TestJava8 {
 
  public static void main(String[] args) {
    String[] array = {"a", "b", "c", "d", "e"};
     Supplier<Stream<String>> streamSupplier = () -> Stream.of(array);
 
    //get new stream
    streamSupplier.get().forEach(x -> System.out.println(x));
 
    //get another new stream
    long count = streamSupplier.get().filter(x -> "b".equals(x)).count();
    System.out.println(count);
  }
 }

Output

a

b

c

d

e

1

每個get()都會返回一個新的流

以上這篇JVM系列之:JIT中的Virtual Call接口操作就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持服務(wù)器之家。

原文鏈接:https://blog.csdn.net/superfjj/article/details/107778750

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 亚洲国产美女精品久久 | 天天干夜夜添 | 久久国产36精品色熟妇 | hh99me福利毛片 | 国产高清视频免费最新在线 | 办公室恋情在线 | 青久草视频| 成人青青草| 韩国激情网 | 国语在线 | 经典千人斩一区二区视频 | 欧美洲大黑香蕉在线视频 | 日本加勒比无码av | 九色PORNY蝌蚪视频首页 | 亚洲成人视屏 | 精品国产乱码久久久久久人妻 | 国产成人啪精品午夜在线观看 | 亚洲熟区 | 无遮免费网站在线入口 | 2012中文字幕中字视频 | 法国女佣系列在线播放 | 欧美成人免费观看国产 | 欧美日韩国产超高清免费看片 | 动漫美女被吸乳 | www.久久99 | 天天黄视频 | 唯美 清纯 另类 亚洲制服 | tube69xxxxhd日本| 亚洲AV无码专区国产乱码网站 | 痴mu动漫成年动漫在线观看 | 国产网站免费观看 | 欧美骚熟 | 亚洲成色www久久网站 | 亚洲国产区中文在线观看 | 青青草99热这里都是精品 | 精品免费视频 | 日韩成人在线视频 | 欧洲一级 | 天天曰天天干 | 91精品国产91热久久久久福利 | gogort人体的最新网站 |