zoukankan      html  css  js  c++  java
  • Java中static块本质论(转载自(http://blog.csdn.net/darxin/article/details/5293427))

    看看下面这段程序输出什么?

    1. public class Test { 
    2.     static
    3.         _i = 20
    4.     } 
    5.     publicstaticint _i = 10
    6.      
    7.     publicstaticvoid main(String[] args) { 
    8.         System.out.println(_i); 
    9.     } 

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

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

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

     
    1. public class Test { 
    2.     public static int _i; 
    3.     static
    4.         _i = 10
    5.     } 

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

    1. public class Test { 
    2.     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."<init>":()V  

       4: return

       static {};  

       Code:   

       0: bipush 10   

       2: putstatic #2; //Field _i:I   

       5: return

    }

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

    1. public class Test { 
    2.     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."<init>":()V   

       4: return

    }

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

    1. public class Test { 
    2.     public static int _i = init(); 
    3.      
    4.     private static int init() { 
    5.         return 10
    6.     } 

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

    1. publicclass Test { 
    2.     public static int _i; 
    3.     static
    4.         _i = init(); 
    5.     } 
    6.      
    7.     private static int init() { 
    8.         return 10
    9.     } 

    问题2:JDK如何处理static块

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

     
    1. public class Test { 
    2.     public static int _i; 
    3.     static
    4.         _i = 10
    5.     } 
    6.      
    7.     public static void main(String[] args) { 
    8.     } 
    9.      
    10.     static
    11.         _i = 20
    12.     } 

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

    public class Test extends java.lang.Object{

       public static int _i;

       public Test();  

       Code:   

       0: aload_0   

       1: invokespecial #1; //Method java/lang/Object."<init>":()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{}部分可以看出,上例的代码与下面的代码效果一致:

    1. public class Test { 
    2.     public static int _i; 
    3.      
    4.     public static void main(String[] args) { 
    5.     } 
    6.      
    7.     static
    8.         _i = 10
    9.         _i = 20
    10.     } 

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

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

    静态变量存放在常量池之中。证明略

    在此需要注意的是:

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

    如:

    1. public class Test { 
    2.     public static int _i = 10

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

    1. public class Test { 
    2.     // 静态变量的声明 
    3.     public static int _i; 
    4.  
    5.     // 静态变量的初始化 
    6.     static
    7.         _i = 10
    8.     } 

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

    1. public class Test { 
    2.     // 静态变量的初始化 
    3.     static
    4.         _i = 10
    5.     } 
    6.      
    7.     // 静态变量的声明 
    8.     public static int _i; 

    对初始问题的解答

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

    1. public class Test { 
    2.     static
    3.         _i = 20
    4.     } 
    5.     public static int _i = 10
    6.      
    7.     public static void main(String[] args) { 
    8.         System.out.println(_i); 
    9.     } 

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

     public class Test { 
    1.     static
    2.         _i = 20
    3.     } 
    4.     public static int _i; 
    5.     static
    6.         _i = 10
    7.     } 
    8.      
    9.     public static void main(String[] args) { 
    10.         System.out.println(_i); 
    11.     } 

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

    1. public class Test { 
    2.     public static int _i; 
    3.      
    4.     static
    5.         _i = 20
    6.         _i = 10
    7.     } 
    8.      
    9.     public static void main(String[] args) { 
    10.         System.out.println(_i); 
    11.     } 

    至此,代码已经明确告诉我们打印结果是什么了!

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

    Result: 10

  • 相关阅读:
    【阿里聚安全·安全周刊】苹果证实 iOS 源代码泄露|英国黑客赢下官司
    150万元重奖!阿里软件供应链安全大赛正式启动
    【阿里聚安全·安全周刊】山寨外挂有风险养蛙需谨慎|健身追踪热度图爆军事基地位置
    移动APP外挂攻防实战
    阿里云正式上线移动直播问答解决方案,助力APP尽情“撒币”!
    阿里安全资深专家杭特辣评中国网络安全人才之“怪现状”
    【阿里聚安全·安全周刊】Intel芯片级安全漏洞事件|macOS存在漏洞
    阿里聚安全年终盘点|2017互联网安全领域十大话题
    【技术分析】DowginCw病毒家族解析
    独家探寻阿里安全潘多拉实验室,完美越狱苹果iOS11.2.1
  • 原文地址:https://www.cnblogs.com/likai198981/p/2848297.html
Copyright © 2011-2022 走看看