博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
静态变量初始化
阅读量:6317 次
发布时间:2019-06-22

本文共 4708 字,大约阅读时间需要 15 分钟。

hot3.png

public class Test {    static {        _i = 20;    }    public static int _i = 10;        public static void main(String[] args) {        System.out.println(_i);    }}

上述代码会打印出什么结果来呢?10还是20?本文将以此代码为引子,着重讨论一下静态变量的初始化问题。

问题1:静态变量如何初始化

Java类中可以定义一个static块,用于静态变量的初始化。如:

public class Test {    public static int _i;    static {        _i = 10;    }}

当然最常用的初始化静态变量的操作是在声明变量时直接进行赋值操作。如:

public class Test {    public static int _i = 10;}

那么上述两例在本质上有什么区别吗?回答是没有区别。两例代码编译之后的字节码完全一致,通过 “javap -c”查看到的字节码如下:

public class Test extends java.lang.Object{public static int _i; public Test();  Code:   0: aload_0   1: invokespecial #1; //Method java/lang/Object."
":()V 4: return static {}; Code: 0: bipush 10 2: putstatic #2; //Field _i:I 5: return }

通过字节码还可以看出,当类的定义中不含有static块时,编译器会为该类提供一个默认的static块。当然这是在含有静态变量初始化操作的前提下。如果静态变量没有初始化操作,则编译器不会为之提供默认的static块。如:

public class Test {    public static int _i;}

其字节码的表现形式为:

public class Test extends java.lang.Object{public static int _i; public Test();  Code:   0: aload_0   1: invokespecial #1; //Method java/lang/Object."
":()V 4: return }

由于静态变量是通过赋值操作进行初始化的,因此可以通过静态函数返回值的方式为其初始化。如:

public class Test {    public static int _i = init();        private static int init() {        return 10;    }}

其本质与下面的代码相同:

public class Test {    public static int _i;    static {        _i = init();    }        private static int init() {        return 10;    }}

问题2:JDK如何处理static块

类定义中可以存在多个static块吗?回答是可以。如:

public class Test {    public static int _i;    static {        _i = 10;    }        public static void main(String[] args) {    }        static {        _i = 20;    }}

此类编译之后的字节码为:

public class Test extends java.lang.Object{public static int _i; public Test();  Code:   0: aload_0   1: invokespecial #1; //Method java/lang/Object."
":()V 4: return public static void main(java.lang.String[]); Code: 0: return static {}; Code: 0: bipush 10 2: putstatic #2; //Field _i:I 5: bipush 20 7: putstatic #2; //Field _i:I 10: return }

观察static{}部分可以看出,上例的代码与下面的代码效果一致:

public class Test {    public static int _i;        public static void main(String[] args) {    }        static {        _i = 10;        _i = 20;    }}

此例可以证明,不仅类定义中可以有多个static块,而且在编译时编译器会将多个static块按照代码的前后位置重新组合成一个static块。

问题3:如何看待静态变量的声明

静态变量存放在常量池之中。如何证明呢?如:

public class Test {    public static int _i = 10;}

使用“javap -c -verbose”查看其字节码的内容如下:

public class Test extends java.lang.Object  SourceFile: "Test.java"  minor version: 0  major version: 49  Constant pool:const #1 = Method #4.#14; //  java/lang/Object."
":()Vconst #2 = Field #3.#15; // Test._i:Iconst #3 = class #16; // Testconst #4 = class #17; // java/lang/Objectconst #5 = Asciz _i;const #6 = Asciz I;const #7 = Asciz
;const #8 = Asciz ()V;const #9 = Asciz Code;const #10 = Asciz LineNumberTable;const #11 = Asciz
;const #12 = Asciz SourceFile;const #13 = Asciz Test.java;const #14 = NameAndType #7:#8;// "
":()Vconst #15 = NameAndType #5:#6;// _i:Iconst #16 = Asciz Test;const #17 = Asciz java/lang/Object; {public static int _i;public Test(); Code: Stack=1, Locals=1, Args_size=1 0: aload_0 1: invokespecial #1; //Method java/lang/Object."
":()V 4: return LineNumberTable: line 2: 0 static {}; Code: Stack=1, Locals=0, Args_size=0 0: bipush 10 2: putstatic #2; //Field _i:I 5: return LineNumberTable: line 3: 0 }

我们看到,常量池中const #2指向的就是Test._i,也就是静态变量。静态变量被保存到常量池中的工作原理这里不深入讨论。在此需要注意的是:

  • 静态变量的声明与初始化是两个不同的操作;
  • 静态变量的声明在编译时已经明确了内存的位置。

如:

public class Test {    public static int _i = 10;}

上述代码的本质可以视为:

public class Test {    // 静态变量的声明    public static int _i;    // 静态变量的初始化    static {        _i = 10;    }}

由于静态变量的声明在编译时已经明确,所以静态变量的声明与初始化在编码顺序上可以颠倒。也就是说可以先编写初始化的代码,再编写声明代码。如:

public class Test {    // 静态变量的初始化    static {        _i = 10;    }        // 静态变量的声明    public static int _i;}

对初始问题的解答

解答了上述三个问题,让我们再来看看开篇提到的问题。代码如下:

public class Test {    static {        _i = 20;    }    public static int _i = 10;        public static void main(String[] args) {        System.out.println(_i);    }}

其本质可以用下面的代码表示:

public class Test {    static {        _i = 20;    }    public static int _i;    static {        _i = 10;    }        public static void main(String[] args) {        System.out.println(_i);    }}

再简化一下,可以表示为:

public class Test {    public static int _i;        static {        _i = 20;        _i = 10;    }        public static void main(String[] args) {        System.out.println(_i);    }}

总结

jvm在加载类的时候,先只是建立的一堆索引而已,类属性的默认值都为0,然后再赋值,不是说你给个private int a = 1;那么a就等于1了,总得有个先后顺序。先建立类的每个属性索引,然后才会赋值。

转载于:https://my.oschina.net/u/2000675/blog/1560327

你可能感兴趣的文章
温瑞尔NFV平台加快高效虚拟CPE部署
查看>>
数据库之触发器
查看>>
大国企纷纷盯上“阿里云”,打造中国的“Predix”
查看>>
视频监控的延伸-视频直播的应用价值探讨
查看>>
android 基础框架依赖库
查看>>
《 测试反模式:有效规避常见的92种测试陷阱》——3.2 一般建议
查看>>
《IPv6安全》——1.6 小结
查看>>
《Hadoop MapReduce实战手册》一1.2 在你的机器上安装Hadoop
查看>>
Java 运算符
查看>>
《C和C++代码精粹》——2.11 更高深的内容
查看>>
《Adobe Flash CS6中文版经典教程》——1.12 发布影片
查看>>
《Android智能穿戴设备开发指南》——第6章,第6.2节使用TCP协议传输数据
查看>>
《C Primer Plus(第6版)中文版》一1.4 计算机能做什么
查看>>
《Ansible权威指南》一2.7 本章小结
查看>>
一些重要 Docker 命令的简单介绍
查看>>
微服务,微架构[六]之springboot集成mybatis
查看>>
RDS SQL Server - 专题分享 - 巧用执行计划缓存之索引缺失
查看>>
诺奖得主Wilczek:人工智能正在解放我们的大脑
查看>>
自定义企业首页文档
查看>>
C++实践参考——分数类中的运算符重载
查看>>