碰见公司神人神代码了,居然写的代码没看懂执行顺序,代码类似如下。
/** * Created by 16. */ class Snippet { static { System.out.println("11"); //System.out.println(a);//compile error: illegal forward reference a = 20;//这里居然不报错。会有人这么写代码? System.out.println("12"); //System.out.println(a);//compile error: illegal forward reference } //private static int a=1; static { System.out.println("13"); //System.out.println(a);//compile error: illegal forward reference } private static int a = 1; public static void main(String[] args) { System.out.println("3 a=" + a); System.out.println("5 a=" + Snippet.a); System.out.println("4 a=" + a); } static{ System.out.println("1 a=" + a); a=10; System.out.println("2 a=" + a); } } /* # 首先遵循原则:先定义后使用。 所以在定义之前使用会报错。见上面。 # 加载类的时候会完成静态初始化操作。 所以在main运行前,需要完成初始化,所以main下面的static会比main先执行。 # 同级别的static,按照顺序执行。 # 注意static中的语句,遵照顺序排列的。所以输出语句 */
输出(原理我写了注释):
11
12
13
1 a=1
2 a=10
3 a=10
5 a=10
4 a=10
----
本篇文章值得一看:Java静态变量的初始化(static块的本质)
在网上看到了下面的一段代码:
- 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."<init>":()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."<init>":()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."<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{}部分可以看出,上例的代码与下面的代码效果一致:
- 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."<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,也就是静态变量。静态变量被保存到常量池中的工作原理这里不深入讨论。在此需要注意的是:
- 静态变量的声明与初始化是两个不同的操作;
- 静态变量的声明在编译时已经明确了内存的位置。
如:
- 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);
- }
- }
至此,代码已经明确告诉我们打印结果是什么了!
+
+
+
=
+
=
=
相关推荐
详细介绍了Java的静态成员变量、静态数据块、非静态成员变量和非静态成员变量等初始化顺序
java 静态_非静态 字段_方法_代码块 子类父类构造_初始化顺序! 三个class 让你清清楚楚 第一个class java代码如下: package initialOrder; class Parent { // 静态变量 public static String p_StaticField...
静态初始化块和对象初始化块的执行顺序不同。静态初始化块在类加载时执行,而对象初始化块在创建新对象时执行。它们可以相互配合,以实现更复杂的初始化逻辑。例如,静态初始化块可以用来设置全局配置,而对象初始化...
特别是当涉及到静态域(静态变量)、非静态域(实例变量)、静态块、非静态块以及构造函数时,明确它们的初始化顺序有助于避免潜在的编程错误。 根据题目提供的内容,我们将重点讨论这些概念以及它们之间的相互关系...
这个顺序表明,无论类之间的继承关系如何,初始化顺序始终是:静态变量和静态初始化块先于非静态成员。在创建对象时,父类的初始化先于子类。这是Java语言规范所规定的,确保在子类访问父类的静态或非静态成员时,...
在本文中,我们讨论了 Java 类加载器中的静态变量初始化机制,了解了静态变量的初始化顺序和类加载器的生命周期。通过对静态变量初始化机制的理解,我们可以更好地掌握 Java 语言的基础知识,并更好地应用 Java 语言...
Java 中的静态变量、静态方法、静态块和静态类 Java 中的静态变量、静态方法、静态块和静态类是 Java 编程语言的四个重要概念,它们之间存在着紧密的关系。下面将对这四个概念进行详细的介绍。 一、静态变量...
以下是对Java程序初始化顺序的详细说明: 1. **类加载阶段**: - **加载**:当Java虚拟机(JVM)首次遇到一个类的引用时,会通过类加载器进行加载。加载过程包括找到类的.class文件,读取其字节码,并转化为内存中...
Java 变量初始化的时机是指在 Java 语言中变量的初始化过程,包括变量的声明、初始化和赋值的步骤。 Java 变量声明 在 Java 语言中,每个变量必须先声明,然后才能使用。声明变量时必须指定该变量的数据类型。...
同时,掌握类变量初始化顺序可以避免因误解而导致的错误,特别是在多线程环境中,对静态变量的并发访问和初始化顺序的控制需要特别注意。 总之,深入理解Java的ClassLoader机制和类变量初始化顺序是提升Java编程...
例如,在 `Bowl` 类中,`b6` 和 `b9` 是两个静态变量,它们的初始化顺序是按照它们在类中的定义顺序进行的。 在 `main` 函数中,我们可以看到创建了两个 `Cupboard` 对象,每个对象的创建都会触发静态变量的初始化...
在 JAVA 中,类的初始化顺序可以分为四个阶段:静态变量、静态初始化块、变量、初始化块和构造器。其中,静态变量和静态初始化块的初始化顺序是最高的,接着是变量和初始化块,最后是构造器。 在了解类的初始化顺序...
### Java中类的初始化顺序详解 #### 一、概述 在Java编程语言中,类的初始化是一个非常重要的概念。类的初始化涉及到多个方面,包括静态成员变量、实例成员变量、静态初始化块、实例初始化块以及构造函数等。本文...
总之,Java代码的初始化顺序是类加载的必然过程,涉及到静态和实例初始化块、构造函数、成员变量初始化以及继承关系的影响。这个demo是学习和理解这些概念的重要工具,通过实际操作可以加深对Java内存管理和对象生命...
总之,Java类继承初始化顺序涉及到静态和非静态初始化块、构造器的调用以及方法的覆盖。理解这些概念对于编写健壮的、易于维护的Java代码至关重要。在实际编程中,应合理利用继承特性,同时注意避免不必要的复杂性和...
首先,需要了解Java类初始化的基本规则,即在类的静态变量、静态初始化块、变量、初始化块、构造器这五个部分中,它们的初始化顺序依次是:静态变量、静态初始化块、变量、初始化块、构造器。这个顺序在单个类中是...
Java编程语言中,静态块(static block)和初始化块(instance initialization block)是两种特殊的代码块,它们在程序运行的不同阶段被执行,对于类和对象的初始化有着重要作用。同时,`main`方法是Java程序的入口...
本篇文章将围绕“类继承的初始化顺序”这一主题展开,详细解析初始化过程中涉及的关键概念和技术细节。 ### 类继承的初始化顺序概述 在面向对象语言中(如Java、C#等),当创建一个继承自某个基类的子类对象时,会...
通过这两个示例,我们可以看到,非静态成员变量的初始化顺序是:成员变量初始化语句>成员变量初始化块>构造函数。 而对于静态成员变量,它们是在类加载时初始化的。例如,我们可以定义一个静态成员变量static ...
Java语言中的类初始化顺序是面试中常见的问题,尤其对于Java程序员和工程师来说,理解这一概念至关重要。本篇文章将深入解析类初始化的顺序以及在继承情况下的表现。 首先,我们需要明确类初始化顺序的基本规则: ...