一,静态类
二,静态方法
[访问权限修饰符] static [返回值类型] 方法名([参数列表]){
语句序列
}
在外部调用静态方法时,可以使用“类名.方法名”的方式,也可以使用“对象名.方法名”的方式。
静态方法在访问本类的成员时,只允许访问静态成员,而不允许访问实例成员变量和实例方法。
三,静态代码块
static {
语句序列
}
静态代码块只能定义在类里面,不能定义在方法里面。静态代码块里面定义的变量都是局部变量,只在本块内有效。
静态代码块会在类被加载时(New 新的对象或者首次使用类名调用静态方法eg:static main()时)自动执行,而无论加载者是JVM还是其他的类。
一个类中允许定义多个静态代码块,执行的顺序根据定义的顺序进行。
静态代码块只能访问类的静态成员,而不允许访问实例成员。
在main方法之前就被执行
四,静态成员变量
如果是基本数据类型的静态成员变量,在类的外部可以不必创建对象就直接使用
系统不会为静态的类变量创建实例
public class supplyTest{
public static void statShow(){
System.out.println("这是静态方法");
}
public void instShow(){
System.out.println("这是实例方法");
}
}
public class hasStatMember{
static supplyTest stVar;
public static void main(String args[]){
stVar.statShow();
stVar.instShow();
}
}
可以编译通过,运行结果如下:这是静态方法
Exception in thread "main" java.lang.NullPointerException
at hasStatMember.main(hasStatMember.java:5)
从运行结果中可以看出,静态方法被正常执行,但实例方法不能执行,原因是未创建对象实例。这说明尽管stVar被声明成static类型,系统仍然不会自动为它创建对象,所以程序必须改成如下内容才能正常运行:
public class hasStatMember{
static supplyTest stVar = new supplyTest();
public static void main(String args[]){
stVar.statShow();
stVar.instShow();
}
}
程序的输出结果是:
这是静态方法
这是实例方法
从输出结果中可以看出,stVar的实例化是在定义时完成的,这意味着在hasStatMember类的外部可以像在内部一样使用它。下面这个程序演示了对stVar的使用形式。
public class useStVar{
public static void main(String args[]){
hasStatMember.stVar.statShow();
hasStatMember.stVar.instShow();
}
}
程序的输出结果如下:
这是静态方法
这是实例方法
无论是静态方法还是实例方法,都是通过“类名.静态变量名.方法名”的形式来使用的。读者可能会觉得这种形式有点眼熟。确实如此,前面大量使用的“System.out.println”就是这种形式。其中,System是系统预定义好的一个类,out是它的一个静态成员,println是out的一个实例方法。
五,Java中的初始化顺序
JAVA类首次装入时,会对静态成员变量或方法进行一次初始化,但方法不被调用是不会执行的,静态成员变量和静态初始化块级别相同,非静态成员变量和非静态初始化块级别相同。
初始化顺序:先初始化父类的静态代码--->初始化子类的静态代码-->
(创建实例时,如果不创建实例,则后面的不执行)初始化父类的非静态代码(变量定义等)--->初始化父类构造函数--->初始化子类非静态代码(变量定义等)--->初始化子类构造函数
类只有在使用New调用创建的时候才会被JAVA类装载器装入创建类实例时,首先按照父子继承关系进行初始化类实例创建时候,首先初始化块部分先执行,然后是构造方法;然后从本类继承的子类的初始化块执行,最后是子类的构造方法类消除时候,首先消除子类部分,再消除父类部分