1. 类变量的初始化可通过静态初始化块来进行。

代码放在一对大括号内,大括号前用static修饰:static {……}

一个类可定义1个或多个静态初始化块。

  静态初始化块会在加载时调用而且只被调用一次。

属于类的公共域,为该类所有对象共享。so它可看做是类范围内的一种的全局变量。

静态域的值由静态方法改变。

class A {
public A() {System.out.println("A's constructor!");}
}
class TestStaticBlock extends A {
//constructor
public TestStaticBlock() {
this("second");
System.out.println("begin constructor");
System.out.println(s_a);
System.out.println(s_b);
System.out.println(c);
System.out.println(d);
// this("second");//call to this must be first statement in constructor
s_a = 1111;
s_b = 2222;
c = 3333;
d = 4444;
System.out.println(s_a);
System.out.println(s_b);
System.out.println(c);
System.out.println(d);
System.out.println("end constructor");
}
//second constructor
public TestStaticBlock(String s) {
System.out.println("begin second constructor");
System.out.println("end second constructor");
}
//main
public static void main(String args[]) {
System.out.println("begin main");
System.out.println(s_a);
System.out.println(s_b);
// System.out.println(c);//non-static variable c cannot be referenced
// from a static context
// System.out.println(d);//non-static variable c cannot be referenced
// from a static context
s_a = 11111;
s_b = 22222;
// c=33333;//non-static variable c cannot be referenced from a static
// context
// d=44444;//non-static variable c cannot be referenced from a static
// context
System.out.println(s_a);
System.out.println(s_b);
// System.out.println(c);//non-static variable c cannot be referenced
// from a static context
// System.out.println(d);//non-static variable c cannot be referenced
// from a static context
System.out.println("before new class object");
TestStaticBlock t = new TestStaticBlock();
System.out.println("end new class object");
System.out.println(s_a);
System.out.println(s_b);
// System.out.println(c);//non-static variable c cannot be referenced
// from a static context
// System.out.println(d);//non-static variable c cannot be referenced
// from a static context
s_a = 111111;
s_b = 222222;
// c=333333;//non-static variable c cannot be referenced from a static
// context
// d=444444;//non-static variable c cannot be referenced from a static
// context
System.out.println(s_a);
System.out.println(s_b);
// System.out.println(c);//non-static variable c cannot be referenced
// from a static context
// System.out.println(d);//non-static variable c cannot be referenced
// from a static context
System.out.println("end main");
}
//define static s_a
static int s_a = 1;
//define c
int c = 3; //block
{
System.out.println("begin block");
System.out.println(s_a);
System.out.println(s_b);
System.out.println(c);
// System.out.println(d);//illegal forward reference
s_a = 111;
s_b = 222;
c = 333;
d = 444;
System.out.println(s_a);
System.out.println(s_b);
System.out.println(c);
// System.out.println(d);//illegal forward reference
System.out.println("end block");
} //static block
static {
System.out.println("begin static block");
System.out.println(s_a);
// System.out.println(s_b);//illegal forward reference
// System.out.println(c);//non-static variable c cannot be referenced
// from a static context
// System.out.println(d);//non-static variable d cannot be referenced
// from a static context
s_a = 11;//这个s_a之前定义过
s_b = 22; //这个是在下面的*处定义的,但是为什么不出错??
System.out.println(s_a);
// System.out.println(s_b);//illegal forward reference定义该字段之前不能引用该字段
// System.out.println(c);//non-static variable c cannot be referenced
// from a static context
// System.out.println(d);//non-static variable c cannot be referenced
// from a static context
System.out.println("end static block");
}
//define d
int d = 4;
//define static s_b
static int s_b = 2;
}

输出结果:  

begin static block
1
11
end static block
begin main
11
2
11111
22222
before new class object
A's constructor!
begin block
11111
22222
3
111
222
333
end block
begin second constructor
end second constructor
begin constructor
111
222
333
4
1111
2222
3333
4444
end constructor
end new class object
1111
2222
111111
222222
end main

通过对输出进行分析,可以得出如下结果: 
1、在类第一次加载时候,会执行静态域(field)初始化语句和静态块(用static{}包含的部分)。 
这里要注意: 
    a、不管静态域声明语句的实际位置在哪儿,当第一次加载类的时候都会首先对它初始化为缺省值(0,false,null等)。 
    b、即使静态域声明中使用了显式初始化语句(比如:int x=3),第一次加载类的时候也会先把它初始化为缺省值(此时x为0),然后再按照下面说的要点c来执行赋值语句(x=3)。 
    c、对于静态域的显式初始化语句和静态块,按照在类中代码出现的先后顺序执行。 
     因此,在上面的例子程序中,我们看到 
      static int s_a=1; 
      static 
      { 
         s_a=11; 
         s_b=22; 
       } 
       static int s_b=2; 
      对s_a,s_b会有不同的效果。类加载时候,s_a,s_b都被初始化为0,然后由于依照代码顺序执行了s_a=1;s_a=11;s_b=22;s_b=2;结果s_a、s_b分别变成了11和2。

2、当构造类实例时候,会先对实例域初始化为缺省值,然后执行实例块(用{}括起来的部分),然后执行构造方法。其中: 
    a、如同1中一样,如果有实例域的显式初始化语句,程序仍然是先将该域初始化为缺省值,然后按照代码在类中出现的先后顺序执行初始化语句或者实例块。如果实例块位置在初始化语句前面,即使它改变了该域的值,也会被随后执行的初始化语句改回去。 
    b、在进入构造方法后,如果构造方法第一句是使用this(...)调用另一构造方法的话,则先执行另一构造方法,然后再执行本构造方法的方法体。这种用法必须让this(...)位于第一句。

《Core java 2》书中所说的"进入构造方法后,如果第一句是调用别的构造方法,则进入别的构造方法。否则,执行实例块"的提法有问题。事实是,不管是否使用this()都会先执行实例块,再进入构造方法。另外,本程序需要在sdk1.4下编译,在sdk1.3下编译将不允许在静态块或实例块中改变位置在它们后面声明的域的值。(看下面的例子,可以知道,core java 2中说得没错,进入构造方法,没错.如果第一句是调用别的构造方法(包括因为继承了父类而有个隐形的构造方法),则就进入别的构造方法,否则,先执行实例块,再回到构造方法里执行往下的内容.事实上已经进入了构造方法之后,先执行实例块,再执行构造方法.而不是反驳者所说的:"事实是,不管是否使用this()都会先执行实例块,再进入构造方法。"

在看一个例子:

public class TestExtend4 {
public static void main(String[] args) {
C c = new C(); //1
}
}
class A {
static {
System.out.println("1"); //2
}
public A() { //7
System.out.println("2"); //8
}
} class B extends A {
static {
System.out.println("a"); //3
}
public B() { //6
System.out.println("b"); //9
}
} class C extends B {
static {
System.out.println("I"); //4
}
public C() { //5
System.out.println("II"); //11
} {
System.out.println("yeah!"); //10
}
}

  结果:
1
a
I
2
b
yeah!
II

在eclipse中用debug运行可以看得出这段程序的有标有断点的语句的执行顺序,如上面序号所示.可以总结出某些继承和静态的知识:

1.当要载入类时(注意只会载入一次),先初始化类里的静态域声明变量为默认值,(即使是赋值了,int值也要为默认的0)然后按顺序的执行静态域和静态块.
2.但是当此类有父类时,要先如果父类有静态域或者静态块,先执行父类的,再执行子类的.
3.当执行完有关静态的东西时,就到了构造类实例时候,会先对实例域初始化为缺省值,然后执行实例块(用{}括起来的部分),然后执行构造方法。
4.但是当此类有父类时,就应该先进入父类的构造方法执行内容,然后再回到子类构造方法里,然后再跳出子类构造方法,执行子类里的实例域和实例块,然后再回到子类构造方法里,执行子类构造方法里面的内容.(就像上面所说的:《Core java 2》书中所说的"进入构造方法后,如果第一句是调用别的构造方法,则进入别的构造方法。否则,执行实例块")意思是无论其是不是继承了父类,还是调用了自己类的另一个构造方法,都会先执行,否则,如果有实例块就先执行实例块.最后再执行构造方法里面的内容.

Thinking in Java 第七章 里面的例子:

class Meal {
Meal() {
System.out.println("Meal()");
}
}
class Bread {
Bread() {
System.out.println("Bread()");
}
}
class Cheese {
Cheese() {
System.out.println("Cheese()");
}
}
class Lettuce {
Lettuce() {
System.out.println("Lettuce()");
}
}
class Lunch extends Meal {
Lunch() {
System.out.println("Lunch()");
}
}
class PortableLunch extends Lunch {
PortableLunch() {
System.out.println("PortableLunch()");
}
}
public class Sandwich extends PortableLunch {
private Bread b = new Bread();
private Cheese c = new Cheese();
private Lettuce l = new Lettuce();
public Sandwich() {
System.out.println("Sandwich()");
} {
System.out.println("yeah");
}
public static void main(String[] args) {
new Sandwich();
}
}

 输出结果: 

Meal()
Lunch()
PortableLunch()
Bread()
Cheese()
Lettuce()
yeah
Sandwich()

说明: Sandwich是public主类,反映三层级别的继承(若将从Object的隐含继承也算在内,就是四级),构造此复杂对象调用构造器要遵照下面的顺序:
1.调用基类构造器.这个步骤会不断地反复递归下去,首先是构造这种层次结构的根,然后是下一层导出类,等等.直到最底层的导出类.
2.按声明顺序调用成员的初始状态设置模块(先初始化类成员变量为默认值(即使有设置值也要设置成默认值,譬如int x = 3;的话,x初始化为0.),然后按照代码顺序执行实例域和实例块)
3.最后执行此类构造方法里面的内容.

最后说明一点:

静态初始化器对静态域进行初始化,构造方法也是完成初始化工作。但二者本质不同,静态初始器是类进入内存时,系统调用执行,而构造方法是执行new后自动执行。

java静态初始化块(静态域)的更多相关文章

  1. java静态初始化块的执行顺序

    先来观察下面的代码 package trr; class Root { static{ System.out.println("Root的静态初始化块"); } { System. ...

  2. Java 静态初始化块等的执行顺序

    实例代码 package text; class Root { static{ System.out.println("Root的静态初始化块"); } { System.out. ...

  3. Java的初始化块、静态初始化块、构造函数的执行顺序及用途探究

    Java与C++有一个不同之处在于,Java不但有构造函数,还有一个”初始化块“(Initialization Block)的概念.下面探究一下它的执行顺序与可能的用途. 执行顺序 首先定义A, B, ...

  4. Java 中的 static 使用之静态初始化块

    Java 中可以通过初始化块进行数据赋值.如: 在类的声明中,可以包含多个初始化块,当创建类的实例时,就会依次执行这些代码块.如果使用 static 修饰初始化块,就称为静态初始化块. 需要特别注意: ...

  5. java中的静态初始化块

    Java 中可以通过初始化块进行数据赋值.如: 在类的声明中,可以包含多个初始化块,当创建类的实例时,就会依次执行这些代码块.如果使用 static 修饰初始化块,就称为静态初始化块. 需要特别注意: ...

  6. java中初始化块、静态初始化块和构造方法

    (所谓的初始化方法init()是另一回事, 在构造方法之后执行, 注意不要混淆) 在Java中,有两种初始化块:静态初始化块和非静态初始化块.它们都是定义在类中,用大括号{}括起来,静态代码块在大括号 ...

  7. Java类的初始化顺序 (静态变量、静态初始化块、变量、初始...

    很有意思的一篇文章 1.没有继承 静态变量->静态初始化块->变量->变量初始化块->构造方法 2.有继承的情况 父类静态变量->父类静态初始化块->子类静态变量- ...

  8. 浅谈Java中静态初始化块跟非初始化块

    众所周知在JAVA编程语言中有两种初始化块:   静态初始化块 非静态初始化块 他们到底有什么区别呢?今天就浅谈一下JAVA中静态初始化块和非静态初始化块的区别   静态初始化块 定义:       ...

  9. java中静态初始化块的执行顺序

    在java中,其应该是先于所有的方法执行. 下面是测试代码: public class Test1 { static{ System.out.println("执行静态初始化块test1.. ...

随机推荐

  1. 一篇文章让你彻底理解java中抽象类和接口

    目录 1.我所理解的抽象类 2.我所理解的接口 3.抽象类和接口本质区别 相信大家都有这种感觉:抽象类与接口这两者有太多相似的地方,又有太多不同的地方.往往这二者可以让初学者摸不着头脑,无论是在实际编 ...

  2. mysql如何解除死锁状态

    第一种: 1.查询是否锁表 show OPEN TABLES where In_use > 0; 2.查询进程(如果您有SUPER权限,您可以看到所有线程.否则,您只能看到您自己的线程) sho ...

  3. scrapy爬取京东iPhone11评论(一)

    咨询行业中经常接触到文本类信息,无论是分词做词云图,还是整理编码分析用,都非常具有价值. 本文将记录使用scrapy框架爬取京东IPhone11评论的过程,由于一边学习一边实践,更新稍慢请见谅. 1. ...

  4. 基于verdaccio的npm私有仓库搭建

    详见个人博客:https://shengchangwei.github.io/verdaccio/ 一.使用npm安装 npm install --global verdaccio 二.cmd 启动 ...

  5. Map Reduce 论文阅读

    Map Reduce 是 Google 在 2004 年发表的一篇论文,原文链接 在这 后来 Hadoop 直接内置了这一框架. 读完之后记录一下心得. 主要背景:MapReduce 的出现很具有工程 ...

  6. 【PyTorch教程】P3. Python学习中的两大法宝函数(当然也可以用在PyTorch)

    温馨提示:为了更好的教程体验,提供视频.阅读地址 Youtube: https://www.youtube.com/playlist?list=PLgAyVnrNJ96CqYdjZ8v9YjQvCBc ...

  7. web应用安全框架选型:Spring Security与Apache Shiro

    一. SpringSecurity 框架简介 官网:https://projects.spring.io/spring-security/ 源代码: https://github.com/spring ...

  8. 安利一个绘制指引线的JS库leader-line

    前言 之前看到一篇推荐Magi这个搜索引擎的新闻,对于这个搜索引擎是否好用咱们不予置评,但是我在这个搜索引擎上面发现了一个好玩的前端功能. 如上图,将鼠标浮动到学习来源上时,会展示一堆指引线. 本博客 ...

  9. 130道ASP.NET面试题(一)

    1 .简述 private,protected,public,internal修饰符的访问权限 答: private : 私有成员, 在类的内部才可以访问. protected : 保护成员,该类内部 ...

  10. Python 基础之re 模块

    Python 基础之大话 re 在使用re模块中主要会用到一下几个方法: re.match() #从头匹配一个字符串 re.search() #浏览全部字符串,匹配第一个符合规则的字符串 re.fin ...