美文网首页JVM
Java之常量折叠、常量传播和Global Value Numb

Java之常量折叠、常量传播和Global Value Numb

作者: 三也视界 | 来源:发表于2021-02-26 09:10 被阅读0次

常量折叠

常量折叠是Java在编译期做的一个优化,简单的来说,在编译期就把一些表达式计算好,不需要在运行时进行计算。
比如: int a = 1 + 2,经过常量折叠后就变成了int a = 3
我们举个例子:

public class Main {
    public static void main(String[] args) {
        String s1 = "a" + "bc";
        String s2 = "ab" + "c";
        System.out.println(s1 == s2);
    }
}

执行结果为true。
我们使用javac编译之后,在通过反编译工具(这个有个网站可以用)看下编译器优化后的代码:

public class Main {
   public static void main(String[] var0) {
      String var1 = "abc";
      String var2 = "abc";
      System.out.println(var1 == var2);
   }
}

var1和var2的值都是常量池中的”abc”,是同一个引用,所以会相等。
修改下我们的例子:

public class Main {
    public static void main(String[] args) {
        String a = "a";
        String b = "b";
        String s1= a + b;
        String s2= a + b;
        System.out.println(s1 == s2);
    }
}

执行结果为: false。
我们反编译生成的class文件:

public class Main {

   public static void main(String[] var0) {
      String var1 = "a";
      String var2 = "b";
      String var3 = var1 + var2;
      String var4 = var1 + var2;
      System.out.println(var3 == var4);
   }
}

我们知道,对于字符串进行 a + b的代码,运行中是这样处理的:
String s2 = (new StringBuffer()).append(a).append(b).toString();
所以最终得到的s1和s2是不相等(==)的。
第一个例子就是“常量折叠”,并不是所有的常量都会进行折叠,必须是编译期常量之间进行运算才会进行常量折叠,编译器常量就是编译时就能确定其值的常量,这个定义很严格,需要满足以下条件:
1. 字面量是编译期常量(数字字面量,字符串字面量等)。
2. 编译期常量进行简单运算的结果也是编译期常量,如1+2,”a”+”b”。
3. 被编译器常量赋值的 final 的基本类型和字符串变量也是编译期常量。

最后我们举一个final标识的常量折叠的例子:

public class Main {
    static final String a = "a";
    static final String b = "b";
    public static void main(String[] args) {
        String s1= a + b;
        String s2= a + b;
        System.out.println(s1 == s2);
    }
}

输出结果为: true
反编译的结果如下:


public class Main {

   static final String a = "a";
   static final String b = "b";

   public static void main(String[] var0) {
      String var1 = "ab";
      String var2 = "ab";
      System.out.println(var1 == var2);
   }
}

常量折叠,故名思议,在编译优化时,多个变量进行计算时,而且能够直接计算出结果,那么变量将由常量直接替换。如:

 void main()
{
      int a = 3+1-1*5;
      printf("%d",a);
}

优化为:

void main()
{
     printf("%d",-1);
}

常量传播

void main()
 {
     int a = 1;
     printf("%d",a);
 }

编译器在进行编译的时候,将a直接由1代替。

优化后如下:

 void main()
{
     printf("%d",1);
}

Global Value Numbering

Global Value Numbering(GVN) 是一种因为Sea-of-Nodes变得非常容易的优化技术 。

GVN是指为每一个计算得到的值分配一个独一无二的编号,然后遍历指令寻找优化的机会,它可以发现并消除等价计算的优化技术。如果一段程序中出现了多次操作数相同的乘法,那么即时编译器可以将这些乘法合并为一个,从而降低输出机器码的大小。如果这些乘法出现在同一执行路径上,那么GVN还将省下冗余的乘法操作。在Sea-of-Nodes中,由于只存在值的概念,因此GVN算法将非常简单:即时编译器只需判断该浮动节点是否与已存在的浮动节点的编号相同,所输入的IR节点是否一致,便可以将这两个浮动节点归并成一个。比如下面这段代码:

GVN

a = 1;
b = 2;
c = a + b;
d = a + b;
e = d;

GVN会利用Hash算法编号,计算a = 1时,得到编号1,计算b = 2时得到编号2,计算c = a + b时得到编号3,这些编号都会放入Hash表中保存,在计算d = a + b时,会发现a + b已经存在Hash表中,就不会再进行计算,直接从Hash表中取出计算过的值。最后的e = d也可以由Hash表中查到而进行复用。

可以将GVN理解为在IR图上的公共子表达式消除(Common Subexpression Elimination,CSE)。两者区别在于,GVN直接比较值的相同与否,而CSE是借助词法分析器来判断两个表达式相同与否。

对于更详细的编译优化请看
JVM C1 编译优化:合并相同的表达式-Global Value Numbering 之实现
javac 编译与 JIT 编译

相关文章

  • Java之常量折叠、常量传播和Global Value Numb

    常量折叠 常量折叠是Java在编译期做的一个优化,简单的来说,在编译期就把一些表达式计算好,不需要在运行时进行计算...

  • 编译优化算法

    参考资料: 1 编译器常用优化方法 常量传播 将能够计算出结果的变量直接替换为常量 优化后 常量折叠 多个变量计算...

  • 第一章 Kotlin基础语法

    一、常量与变量(val,var) 1.什么是常量? 1 .val = value ,值类型;2.类似Java的fi...

  • 常量折叠

    常量折叠说的是,在编译阶段,对该变量进行值替换,同时,该常量拥有自己的内存空间,并非像宏定义一样不分配空间。 示例...

  • java__常量池

    java的常量池分为两种型态:静态常量池和运行常量池 静态常量池: 即class文件中的常量池,这种常量池主要用于...

  • Java常量、变量

    Java常量的定义和分类 常量值 整型常量值Java 的整型常量值主要有如下 3 种形式。 十进制数形式:如 54...

  • 常量

    常量  Java中由final修饰的就是常量。如下:  分类:编译期常量和运行时常量 编译期常量:它的值在编译期就...

  • java基础类型、String类理解、版本对比、1.8新特性

    1、java基本数据类型及长度 2、jvm的常量池: JVM常量池浅析Java常量池理解与总结 Java中的常量池...

  • 自定义全局常量,类常量,类静态属性

    define('name', 'value') 自定义全局常量,默认大小写敏感const 定义类常量, 常量明前不...

  • Java基础语法之常量

    1.Java中常量分类 字面值常量 自定义常量 2.字面值常量的分类 字符串常量 整数常量 小数常量 字符常量 布...

网友评论

    本文标题:Java之常量折叠、常量传播和Global Value Numb

    本文链接:https://www.haomeiwen.com/subject/hetufltx.html