Java中的静态变量、静态方法与静态代码块(转载)

我们知道类的生命周期分为装载、连接、初始化、使用和卸载的五个过程。

  其中静态代码在类的初始化阶段被初始化。而非静态代码则在类的使用阶段(也就是实例化一个类的时候)才会被初始化。

  静态变量

  可以将静态变量理解为类变量(与对象无关),而实例变量则属于一个特定的对象。

  静态变量有两种情况:

  ● 静态变量是基本数据类型,这种情况下在类的外部不必创建该类的实例就可以直接使用

  ● 静态变量是一个引用。这种情况比较特殊,主要问题是由于静态变量是一个对象的引用,那么必须初始化这个对象之后才能将引用指向它。因此如果要把一个引用定义成static的,就必须在定义的时候就对其对象进行初始化。

public class TestForStaticObject{
    static testObject o = new testObject (); //定义一个静态变量并实例化
    public static void main(String args[]){
    //在main中直接以“类名.静态变量名.方法名”的形式使用testObject的方法
    }
}

静态方法

  与类变量不同,方法(静态方法与实例方法)在内存中只有一份,无论该类有多少个实例,都共用一个方法。

  静态方法与实例方法的不同主要有:

  ● 静态方法可以直接使用,而实例方法必须在类实例化之后通过对象来调用。

  ● 在外部调用静态方法时,可以使用“类名.方法名”或者“对象名.方法名”的形式。实例方法只能使用后面这种方式。

  ● 静态方法只允许访问静态成员。而实例方法中可以访问静态成员和实例成员。

  ● 静态方法中不能使用this、super关键字,因为静态方法在使用前不用创建任何实例对象,当静态方法调用时,this所引用的对象根本没有产生。

  ●静态变量是属于整个类的变量而不是属于某个对象的。注意不能把任何方法体内的变量声明为静态,例如:

fun(){
    static int i=0;//非法。
}

类装载步骤
在Java中,类装载器把一个类装入Java虚拟机中,要经过三个步骤来完成:装载、链接和初始化,其中链接又可以分成校验、准备和解析三步,除了解析外,其它步骤是严格按照顺序完成的,各个步骤的主要工作如下:

装载:查找和导入类或接口的二进制数据;
链接:执行下面的校验、准备和解析步骤,其中解析步骤是可以选择的;
校验:检查导入类或接口的二进制数据的正确性;
准备:给类的静态变量分配并初始化存储空间;
解析:将符号引用转成直接引用;
初始化:激活类的静态变量的初始化Java代码和静态Java代码块。
初始化类中属性是静态代码块的常用用途,但只能使用一次。

 

class Parent{ 
    static String name = "hello"; 
   
   //构造代码块 { System.out.println(
"parent block"); }
   //静态代码块
static { System.out.println("parent static block"); } //构造器 public Parent(){ System.out.println("parent constructor"); } } class Child extends Parent{ static String childName = "hello";
   //构造代码块 { System.out.println(
"child block"); }
   //静态代码块
static { System.out.println("child static block"); }
   //构造器
public Child(){ System.out.println("child constructor"); } } public class StaticIniBlockOrderTest { public static void main(String[] args) { new Child();//语句(*) } }

 

问题:当执行完语句(*)时,打印结果是什么顺序?为什么?
解答:当执行完语句(*)时,打印结果是这样一个顺序 :

parent static block
child static block
parent block
parent constructor
child block
child constructor

 

分析:

当执行new Child()时,它首先去看父类里面有没有静态代码块,如果有,它先去执行父类里面静态代码块里面的内容,当父类的静态代码块里面的内容执行完毕之后, 接着去执行子类(自己这个类)里面的静态代码块,当子类的静态代码块执行完毕之后,它接着又去看父类有没有非静态代码块,如果有就执行父类的非静态代码 块,父类的非静态代码块执行完毕,接着执行父类的构造方法;父类的构造方法执行完毕之后,它接着去看子类有没有非静态代码块,如果有就执行子类的非静态代 码块。子类的非静态代码块执行完毕再去执行子类的构造方法,这个就是一个对象的初始化顺序。

 

总结:
对 象的初始化顺序:首先执行父类静态的内容,父类静态的内容执行完毕后,接着去执行子类的静态的内容,当子类的静态内容执行完毕之后,再去看父类有没有非静 态代码块,如果有就执行父类的非静态代码块,父类的非静态代码块执行完毕,接着执行父类的构造方法;父类的构造方法执行完毕之后,它接着去看子类有没有非 静态代码块,如果有就执行子类的非静态代码块。子类的非静态代码块执行完毕再去执行子类的构造方法。总之一句话,静态代码块内容先执行,接着执行父类非静 态代码块和构造方法,然后执行子类非静态代码块和构造方法。

 

注意:子类的构造方法,不管这个构造方法带不带参数,默认的它都会先去寻找父类的不带参数的构造方法。如果父类没有不带参数的构造方法,那么子类必须用supper关键子来调用父类带参数的构造方法,否则编译不能通过。


转载地址:http://www.cnblogs.com/panjun-Donet/archive/2010/08/10/1796209.html

 

posted @ 2012-10-26 13:41  Ars_blog  阅读(1252)  评论(0编辑  收藏  举报