一.与静态方法的比较
一般情况下,如果有些代码必须在项目启动的时候就执行的时候,需要使用静态代码块,这种代码是主动执行的;需要在项目启动的时候就初始化,在不创建对象的情况下,其他程序来调用的时候,需要使用静态方法,静态方法在类加载的时候 就已经加载 可以用类名直接调用 比如main方法就必须是静态的 这是程序入口。两者的区别就是:静态代码块是自动执行的; 静态方法是被调用的时候才执行的。
二.静态方法注意事项
使用类的静态方法时,注意:
a.在静态方法里只能直接调用同类中其他的静态成员(包括变量和方法),而不能直接访问类中的非静态成员。这是因为,对于非静态的方法和变量,需要先创建类的实例对象后才可使用,而静态方法在使用前
不用创建任何对象。
b.静态方法不能以任何方式引用this和super关键字,因为静态方法在使用前不用创建任何实例对象,当静态方法调用时,this所引用的对象根本没有产生(this关键字只能在方法内部使用,表示对“调用方法的那个对象”的引用)。
静态变量是属于整个类的变量而不是属于某个对象的。注意不能把任何方法体内的变量声明为静态,例如: fun() { static int i=0;//非法。 }
三.程序举例
1 public class TestStaticCon { 2 public static int a = 0; 3 4 static { 5 a = 10; 6 System.out.println("父类的静态代码块在执行a=" + a); 7 } 8 9 { 10 a = 8; 11 System.out.println("父类的非静态代码块在执行a=" + a); 12 } 13 14 public TestStaticCon() { 15 this("a在父类带参构造方法中的值:" + TestStaticCon.a); // 调用另外一个构造方法 16 System.out.println(a); 17 System.out.println("父类无参构造方法在执行a=" + a); 18 } 19 20 public TestStaticCon(String n) { 21 System.out.println(n); 22 System.out.println(a); 23 24 } 25 26 public static void main(String[] args) { 27 TestStaticCon tsc = null; 28 System.out.println("!!!!!!!!!!!!!!!!!!!!!"); 29 tsc = new TestStaticCon(); 30 } 31 }
运行结果: 父类的静态代码块在执行a=10 !!!!!!!!!!!!!!!!!!!!! 父类的非静态代码块在执行a=8 a在父类带参构造方法中的值:10 8 8 父类无参构造方法在执行a=8
四.网友提供
1 public class StaticBlock { 2 3 static { 4 System.out.println("静态块"); 5 } 6 { 7 System.out.println("构造块,在类中定义"); 8 } 9 10 public StaticBlock() { 11 System.out.println("构造方法执行"); 12 } 13 14 public static void main(String[] args) { 15 new StaticBlock(); 16 new StaticBlock(); 17 } 18 19 }
静态块 构造块,在类中定义 构造方法执行 构造块,在类中定义 构造方法执行
五.今天发现的问题
static块不一定非要写在所有变量和方法之前。
public class T2{ private static int total = 200; static{ total = 100; System.out.println("static语句块"); } public static void main(String[] args){ System.out.println(T2.total); System.out.println(T2.total); } }
上面的程序输出的结果是: static语句块 100 100 然后把静态块写在静态变量的前面,编译运行后的结果是 static语句块 200 200 这说明static的加载还是有顺序的 是根据他们在类中的位置加载的; 可以有多个,位置可以随便放,它不在任何的方法体内,JVM加载类时会执行这些静态的代码块,如果static 代码块有多个,JVM将按照它们在类中出现的先后顺序依次执行它们,每个代码块只会被执行一次。
结论:静态代码块是在类加载时自动执行的,非静态代码块是在创建对象时自动执行的代码,不创建对象不执行该类的非静态代码块。且执行顺序为静态代码块------非静态代码块----构造函数。
其中让我疑惑的是“a在父类带参构造方法中的值:10”,我再想那时候为什么不是8,debug了(F11,不能直接设置断点然后运行,那样和直接运行没区别),发现先进入了无参的构造方法,执行了第一条语句并且切换到了另一个构造方法(不管是不是反正第一句都要执行,此时a还是10,非静态代码块还未执行),提示找不到源,不管是不是这条语句都提示了这个警告(不算错误,因为程序继续正常运行),然后运行了非静态代码块,继而从有参的构造方法处继续执行……
今天知道了适应和忍是两种不同的境界……