V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
ukyoo
V2EX  ›  Java

请教一个问题 编译时常量

  •  
  •   ukyoo · 2019-11-19 16:28:27 +08:00 · 1092 次点击
    这是一个创建于 1833 天前的主题,其中的信息可能已经有所发展或是发生改变。

    参考别人的回答 https://www.zhihu.com/question/30292319, static final int 才能作为编译时常量

    RednaxelaFX (作者) 回复尒蜗牛爬呀爬 3 年前 被 final 修饰的当然不“就是常量”。 有 final 修饰的声明中,只有 static final 原始类型或 String 并且带有常量表达式来初始化的那才是 Java 认为的编译时常量。

    但是只给成员变量加上 final 修饰符 final int i = 1111; 反编译.class 文件,发现 int i 也在 CONSTANTPOOL 中

    final int i; descriptor: I flags: ACC_FINAL ConstantValue: int 1111

    7 条回复    2019-11-19 19:23:19 +08:00
    chendy
        1
    chendy  
       2019-11-19 16:39:59 +08:00
    成员变量 final 了还是变量
    静态的基本类型和字符串,加了 final 编译期就作常量替换了
    ukyoo
        2
    ukyoo  
    OP
       2019-11-19 17:31:45 +08:00
    @chendy 请教下, CONSTANT_POOL 里可以看到 final int i = 1111, 不算编译期常量吗
    th00000
        3
    th00000  
       2019-11-19 17:54:50 +08:00
    成员变量 final 了可以增加代码的可读性, 除此之外没有其他作用了。
    InkStone
        4
    InkStone  
       2019-11-19 17:59:36 +08:00
    @ukyoo 可能是编译器优化。

    C++在没有 constexpr 的时候也会做常量传播,但这跟明确的标准语义还是两码事。
    ukyoo
        5
    ukyoo  
    OP
       2019-11-19 18:35:09 +08:00
    @th00000 深入理解 JVM 中提到 Class 的常量池中存放的字面量包括
    字符串
    声明为 final 的常量(并没有提到 static). 比如 final int i = 111,确实存放在 class 的 CONSTANT_POOL 中, 这个算是编译期常量吧
    pdog18
        6
    pdog18  
       2019-11-19 19:22:22 +08:00
    你不加 final 也会出现在 Class 的常量池里面啊。
    我觉得你要把 Class 的常量池和 Java 里面的“常量”给分开
    pdog18
        7
    pdog18  
       2019-11-19 19:23:19 +08:00
    我猜你应该同时看一下
    static final int i = 1111;
    和 final int i = 1111;
    和 int i = 1111;
    在编译成 Class 时的区别。
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   实用小工具   ·   3030 人在线   最高记录 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 27ms · UTC 14:19 · PVG 22:19 · LAX 06:19 · JFK 09:19
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.