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

 更新时间:2020年09月14日 15:44:03   作者:flydean程序那些事  
这篇文章主要介绍了JVM系列之:JIT中的Virtual Call接口操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧

简介

上一篇文章我们讲解了Virtual Call的定义并举例分析了Virtual Call在父类和子类中的优化。

JIT对类可以进行优化,那么对于interface可不可以做同样的优化么?

一起来看看吧。

最常用的接口List

List应该是大家最最常用的接口了,我想这个大家应该不会反驳。

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

今天我们就拿List来做例子,体验一下JIT优化接口的奥秘。

还是上代码,要分析的代码如下:

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.jb51.net");
  }
}

如果在命令行运行,大家记得在运行时添加参数-XX:+UnlockDiagnosticVMOptions -XX:+PrintAssembly -XX:-Inline

直接看JIT Watcher的结果:

我们可以看到JIT中先对ArrayList的实现类做了一个比较。

然后调用的是invokeinterface,但是其本质还是invokevirtual,并且我们可以看到这个调用是被优化过了:optimized virtual call。

多个List的调用

同样的,我们可以测试一下多个list子类的情况下怎么调用:

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.jb51.net");
  }
}

同样,使用JIT Watcher来运行:

我们可以看到JIT做了两次对象类型的比较,然后对两个invokeinterface都做了优化。

结果和我们的父类子类结果是一样的。

不一样的List调用

上面我们在做多个list调用的时候,是轮循着来调用的,如果我们先调用ArrayList的方法,再调用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.jb51.net");
  }
}

上面我们先循环ArrayList,然后再循环LinkedList。

看下结果有什么不同:

可以看到,JIT先比较了ArrayList,然后只做了一次方法的优化。

也就是说LinkedList的调用是没有进行代码优化的。

上面的结果是在C2编译器下,也就是level4的编译水平下解析的。

我们看下如果在C1编译器下,也就是Level3编译水平下有什么不同。

可以看到C1编译下,所有的invokeinterface都没有进行编译优化,只有在C2编译下,才会进行优化。

不同的JVM版本可能优化方式不一样。大家可以自行实验。

总结

本文用实例展示了Virtual Call在interface上面的优化使用。

感兴趣的朋友,可以一起讨论。

补充知识:Java 8 Stream 流已被操作或关闭

在Java 8中,Stream不能重复使用,一旦被消耗或使用,流将被关闭,类似流水线,水龙头的水一样一去不复返

示例 - 流关闭

查看以下示例,它会抛出一个IllegalStateException,表示“流被关闭”。

TestJava8.java

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

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

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接口操作就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • java解析sina视频

    java解析sina视频

    本文介绍了一个java解析sina视频地址的例子,从这个例子中可以学习到java使用sax解析xml的方法,大家可以参考修改成其它功能
    2014-01-01
  • SpringSecurity 自定义表单登录的实现

    SpringSecurity 自定义表单登录的实现

    这篇文章主要介绍了SpringSecurity 自定义表单登录的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-01-01
  • file.mkdir()、file.mkdirs()和file.createNewFile()的区别

    file.mkdir()、file.mkdirs()和file.createNewFile()的区别

    本文主要介绍了file.mkdir()、file.mkdirs()和file.createNewFile()的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04
  • Java结合JS实现URL编码与解码

    Java结合JS实现URL编码与解码

    这篇文章介绍了Java结合JS实现URL编码与解码的方法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-03-03
  • JavaEE Spring MyBatis如何一步一步实现数据库查询功能

    JavaEE Spring MyBatis如何一步一步实现数据库查询功能

    这篇文章主要介绍了JavaEE Spring MyBatis如何一步一步实现数据库查询功能,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-08-08
  • java中利用反射调用另一类的private方法的简单实例

    java中利用反射调用另一类的private方法的简单实例

    下面小编就为大家带来一篇java中利用反射调用另一类的private方法的简单实例。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-06-06
  • JVM:早期(编译期)优化的深入理解

    JVM:早期(编译期)优化的深入理解

    今天小编就为大家分享一篇关于JVM:早期(编译期)优化的深入理解,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-02-02
  • SpringCache快速使用及入门案例

    SpringCache快速使用及入门案例

    Spring Cache 是Spring 提供的一整套的缓存解决方案,它不是具体的缓存实现,本文主要介绍了SpringCache快速使用及入门案例,感兴趣的可以了解一下
    2023-08-08
  • 在SpringBoot中该如何配置拦截器

    在SpringBoot中该如何配置拦截器

    今天给大家带来的是关于SpringBoot的相关知识,文章围绕在SpringBoot中该如何配置拦截器展开,文中有非常详细的介绍及代码示例,需要的朋友可以参考下
    2021-06-06
  • Java实现数据更新和事件通知的观察者模式

    Java实现数据更新和事件通知的观察者模式

    Java观察者模式是一种行为型设计模式,用于实现对象间的一对多依赖关系。当一个对象的状态发生改变时,它的所有依赖对象都会收到通知并自动更新。观察者模式可以实现松耦合,增强了系统的可维护性和可拓展性
    2023-04-04

最新评论