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

请求答疑: Java 中 static final 修改常量的加载问题

  •  
  •   shucc · 2023-10-09 17:09:29 +08:00 · 9131 次点击
    这是一个创建于 447 天前的主题,其中的信息可能已经有所发展或是发生改变。
    public static final int FIRST_LINE_HEIGHT = ScreenUtil.dp2px(80);
    

    我们项目中有以上这一段代码,ScreenUtil.dp2px 即获取 80dp 的高度,存在这个 FIRST_LINE_HEIGHT 数值偶现为 0 的情况,是否是因为 static final 加载时间导致的异常呢?

    14 条回复    2023-10-10 10:11:03 +08:00
    dqzcwxb
        1
    dqzcwxb  
       2023-10-09 17:11:15 +08:00
    ScreenUtil.dp2px()的问题
    TArysiyehua
        2
    TArysiyehua  
       2023-10-09 17:13:04 +08:00
    楼上+1
    chendy
        3
    chendy  
       2023-10-09 17:13:37 +08:00
    默认情况下,静态变量初始化不结束,类加载不结束,外部访问不了这个类
    所以更可能发生的情况是,某些地方用反射破了 final 修改了这个值…
    Ianchen
        4
    Ianchen  
       2023-10-09 17:13:52 +08:00
    典型的 DCL ( double check lock )问题吧?
    lff0305
        5
    lff0305  
       2023-10-09 17:35:18 +08:00   ❤️ 4
    遇到过类似的问题,原因是多个类在 init 的时候互相引用 (有多个 public static final 的 property )。
    例如下面代码打印出来 A 的值是 0

    public class Main {
    public static final int FIRST_LINE_HEIGHT = ScreenUtil.dp2px(80);
    public static void main(String[] argu) {
    }
    }

    class ScreenUtil {
    public static int A = Main.FIRST_LINE_HEIGHT;
    static {
    System.out.println(A);
    }
    public static int dp2px(int i) {
    return i + 1000;
    }
    }
    winterbells
        6
    winterbells  
       2023-10-09 17:38:44 +08:00 via Android
    可能 static 加载的时候 activity 还没启动,设置为-1 ,在 oncreate 里面设置,或者读取的时候判断一下
    cyningxu
        7
    cyningxu  
       2023-10-09 19:37:50 +08:00
    检查 ScreenUtil.dp2px
    oneisall8955
        8
    oneisall8955  
       2023-10-09 20:19:12 +08:00 via Android
    瞎猜一下加个 volitate 防止指令重排?
    lisongeee
        9
    lisongeee  
       2023-10-09 21:13:57 +08:00
    不能像 kotlin 一样使用 lazy 去加载吗?
    silkgrower
        10
    silkgrower  
       2023-10-09 22:53:22 +08:00
    @lff0305 我觉得这层答案对的.跟 Java 加载类的顺序有关.
    q1angch0u
        11
    q1angch0u  
       2023-10-10 00:06:54 +08:00 via iPhone
    static 的变量在类加载阶段 [加载-连接-初始化] 中的连接阶段中的 [准备] 阶段就会做赋值。
    q1angch0u
        12
    q1angch0u  
       2023-10-10 00:08:20 +08:00 via iPhone
    补充一下,final 的 static 变量会在 [准备] 阶段赋值,非 final 的会在 [准备] 阶段赋零值,然后在 [初始化] 阶段的 [clinit] 中赋值。
    chendy
        13
    chendy  
       2023-10-10 08:25:04 +08:00
    @lff0305 静态引用环,恐怖如斯…
    nothingistrue
        14
    nothingistrue  
       2023-10-10 10:11:03 +08:00
    你这个很明显跟 public static final int FIRST_LINE_HEIGH 没关系,跟 ScreenUtil.dp2px(80) 有关系。

    80dp 的高度如果是静态值,那么请直接把它赋值给 FIRST_LINE_HEIGHT 。如果是动态值,那么请不要赋值给 static final 类型的「常量」,给常量设置一个可变值,即使是仅取第一次的值,那也是找抽。
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   实用小工具   ·   1004 人在线   最高记录 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 845ms · UTC 18:48 · PVG 02:48 · LAX 10:48 · JFK 13:48
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.