Java静态变量的初始化(static块的本质)

原创 2010年02月06日 07:23:00

在网上看到了下面的一段代码:

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

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

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

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

那么上述两例在本质上有什么区别吗?回答是没有区别。两例代码编译之后的字节码完全一致,通过 “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块。如:

其字节码的表现形式为:

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

 

}

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

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

问题2:JDK如何处理static块

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

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

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{}部分可以看出,上例的代码与下面的代码效果一致:

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

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

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

使用“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."<init>":()V
const #2 = Field #3.#15; //  Test._i:I
const #3 = class #16; //  Test
const #4 = class #17; //  java/lang/Object
const #5 = Asciz _i;
const #6 = Asciz I;
const #7 = Asciz <init>;
const #8 = Asciz ()V;
const #9 = Asciz Code;
const #10 = Asciz LineNumberTable;
const #11 = Asciz <clinit>;
const #12 = Asciz SourceFile;
const #13 = Asciz Test.java;
const #14 = NameAndType #7:#8;//  "<init>":()V
const #15 = NameAndType #5:#6;//  _i:I
const #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."<init>":()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,也就是静态变量。静态变量被保存到常量池中的工作原理这里不深入讨论。在此需要注意的是:

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

如:

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

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

对初始问题的解答

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

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

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

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

java类中的静态变量是什么时候初始化的

我们知道一个类(class)要被使用必须经过装载,连接,初始化这样的过程。下面先对这三阶段做一个简单的描述,之后会结合一个简单的例子来说明java中类的初始化过程。 在装载阶段,类装载器(Boo...
  • u010585120
  • u010585120
  • 2016年11月09日 10:24
  • 3650

Java静态方法,静态变量,初始化顺序

1. 静态方法:         成员变量分为实例变量和静态变量。其中实例变量属于某一个具体的实例,必须在类实例化后才真正存在,不同的对象拥有不同的实例变量。而静态变量被该类所有的对象公有(相当于全...
  • leeyu35
  • leeyu35
  • 2012年07月17日 14:38
  • 37825

Java类的初始化顺序 (静态变量、静态初始化块、变量、初始化块、构造器)

我们大家都知道,对于静态变量、静态初始化块、变量、初始化块、构造器,它们的初始化顺序以此是(静态变量、静态初始化块)>(变量、初始化块)>构造器。我们也可以通过下面的测试代码来验证这一点:  Jav...
  • caomiao2006
  • caomiao2006
  • 2016年05月29日 19:10
  • 1197

Java类加载器:静态变量初始化

先看两个例子: package wh; class Singleton{ public static Singleton singleton = new Singleton(); publi...
  • woshiwanghao_hi
  • woshiwanghao_hi
  • 2015年11月02日 21:10
  • 1040

关于java中静态变量初始化

题目如下: [java] view plain copy public class Test{       private static Test...
  • u010585120
  • u010585120
  • 2017年04月12日 13:54
  • 294

Java静态变量的初始化(static块的本质)

本文转载自: http://blog.csdn.net/darxin/article/details/5293427 在网上看到了下面的一段代码: [java]...
  • lihongye_10
  • lihongye_10
  • 2013年11月20日 15:05
  • 12008

Java_基础_静态变量一定要先声明后赋值

这标题看着让人很纳闷,什么叫做变量一定要先声明后赋值?Java中的变量不都是先声明后使用的吗?难道还能先使用后声明?能不能暂且不说,我们先来看一个例子,代码如下:package deep; publi...
  • u012552052
  • u012552052
  • 2015年04月10日 09:12
  • 2898

关于Java静态成员变量和静态初始化块等的初始化顺序的详细介绍

  • 2013年09月23日 16:41
  • 7KB
  • 下载

Java静态变量初始化时机

静态变量初始化方式: 1.声明时初始化; 2.静态块中初始化; 测试: package com.skymr.learn; /** * 静态变量初始化顺序测试 * @author skymr * ...
  • naruto_Mr
  • naruto_Mr
  • 2015年06月05日 10:55
  • 708

Java中静态变量和非静态变量的初始化顺序

 先看下面的程序:package c05.net;class Bowl {Bowl(int marker) {System.out.println("Bowl(" + marker + ")");}v...
  • zhu_apollo
  • zhu_apollo
  • 2007年10月28日 15:56
  • 4774
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Java静态变量的初始化(static块的本质)
举报原因:
原因补充:

(最多只允许输入30个字)