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

简介: 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.flydean.com");
    }
}


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


直接看JIT Watcher的结果:


image.png


我们可以看到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.flydean.com");
    }
}


同样,使用JIT Watcher来运行:


image.png


我们可以看到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.flydean.com");
    }
}


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


看下结果有什么不同:


image.png


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


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


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


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


image.png


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


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


总结



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

相关文章
|
2月前
|
存储 缓存 监控
聊聊JIT是如何影响JVM性能的!
聊聊JIT是如何影响JVM性能的!
|
5月前
|
存储 缓存 自然语言处理
(三)JVM成神路之全面详解执行引擎子系统、JIT即时编译原理与分派实现
执行引擎子系统是JVM的重要组成部分之一,在JVM系列的开篇曾提到:JVM是一个架构在平台上的平台,虚拟机是一个相似于“物理机”的概念,与物理机一样,都具备代码执行的能力。
104 1
|
4月前
|
缓存 Java 编译器
JRE、JDK、JVM 和 JIT 之间的区别详解
【8月更文挑战第22天】
197 0
|
5月前
|
Java Windows
Java演进问题之JVM在内存返还策略上会左右为难如何解决
Java演进问题之JVM在内存返还策略上会左右为难如何解决
|
5月前
|
缓存 Java 编译器
Java演进问题之JVMCI JIT编译器与JVM的交互如何解决
Java演进问题之JVMCI JIT编译器与JVM的交互如何解决
|
6月前
|
存储 缓存 Java
JVM的即时编译(JIT)优化原理:加速程序的执行
JVM的即时编译(JIT)优化原理:加速程序的执行
|
7月前
|
监控 安全 Java
JVM工作原理与实战(三十八):JIT即时编译器原理
JVM作为Java程序的运行环境,其负责解释和执行字节码,管理内存,确保安全,支持多线程和提供性能监控工具,以及确保程序的跨平台运行。本文主要介绍了JIT即时编译器、HotSpot中的JIT编译器、JIT优化技术、JIT优化建议等内容。
139 0
|
7月前
|
缓存 监控 Java
jvm的及时编译器JIT
jvm的及时编译器JIT
|
存储 缓存 前端开发
【jvm系列-07】深入理解执行引擎,解释器、JIT即时编译器
【jvm系列-07】深入理解执行引擎,解释器、JIT即时编译器
392 2
|
传感器 Java C#
jvm组成-本地方法接口
了解jvm组成-本地方法接口
112 0