1. 靜態(tài)方法:
成員變量分為實(shí)例變量和靜態(tài)變量。其中實(shí)例變量屬于某一個(gè)具體的實(shí)例,必須在類實(shí)例化后才真正存在,不同的對(duì)象擁有不同的實(shí)例變量。而靜態(tài)變量被該類所有的對(duì)象公有(相當(dāng)于全局變量),不需要實(shí)例化就已經(jīng)存在。
方法也可分為實(shí)例方法和靜態(tài)方法。其中,實(shí)例方法必須在類實(shí)例化之后通過對(duì)象來調(diào)用,而靜態(tài)方法可以在類實(shí)例化之前就使用。與成員變量不同的是:無論哪種方法,在內(nèi)存中只有一份——無論該類有多少個(gè)實(shí)例,都共用同一個(gè)方法。
實(shí)例方法的調(diào)用:
ClassA a = new ClassA(); //必須經(jīng)過實(shí)例化
a.instanceMethod();
靜態(tài)方法的調(diào)用:
a.staticMethod(); //無需經(jīng)過實(shí)例化
2. 靜態(tài)方法的聲明和定義
定義一個(gè)靜態(tài)方法和定義一個(gè)實(shí)例方法,在形式上并沒有什么區(qū)別,只是在聲明的頭部,需要加上一個(gè)關(guān)鍵字static。它的一般語(yǔ)法形式如下:
[訪問權(quán)限修飾符] static [返回值類型] 方法名([參數(shù)列表]){
語(yǔ)句序列
}
例如下面是一個(gè)靜態(tài)的方法:
public static void stFun(){
System.out.println("這是一個(gè)靜態(tài)方法");
}
3.靜態(tài)方法和實(shí)例方法的區(qū)別
靜態(tài)方法和實(shí)例方法的區(qū)別主要體現(xiàn)在兩個(gè)方面:
● 在外部調(diào)用靜態(tài)方法時(shí),可以使用“類名.方法名”的方式,也可以使用“對(duì)象名.方法名”的方式。而實(shí)例方法只有后面這種方式。也就是說,調(diào)用靜態(tài)方法可以無需創(chuàng)建對(duì)象。
● 靜態(tài)方法在訪問本類的成員時(shí),只允許訪問靜態(tài)成員(即靜態(tài)成員變量和靜態(tài)方法),而不允許訪問實(shí)例成員變量和實(shí)例方法;實(shí)例方法則無此限制。
實(shí)例代碼: 靜態(tài)方法訪問成員變量示例
class accessMember{
private static int sa; //定義一個(gè)靜態(tài)成員變量
private int ia; //定義一個(gè)實(shí)例成員變量
//下面定義一個(gè)靜態(tài)方法
static void statMethod(){
int i = 0; //正確,可以有自己的局部變量
sa = 10; //正確,靜態(tài)方法可以使用靜態(tài)變量
otherStat(); //正確,可以調(diào)用靜態(tài)方法
ia = 20; //錯(cuò)誤,不能使用實(shí)例變量
insMethod(); //錯(cuò)誤,不能調(diào)用實(shí)例方法
}
static void otherStat(){
}
//下面定義一個(gè)實(shí)例方法
void insMethod(){
int i = 0; //正確,可以有自己的局部變量
sa = 15; //正確,可以使用靜態(tài)變量
ia = 30; //正確,可以使用實(shí)例變量
statMethod(); //正確,可以調(diào)用靜態(tài)方法
}
}//end of class accessMember
4.靜態(tài)代碼塊
在類中,可以將某一塊代碼聲明為靜態(tài)的,這樣的程序塊叫靜態(tài)初始化段。靜態(tài)代碼塊的一般形式如下:
static {
語(yǔ)句序列
}
● 靜態(tài)代碼塊只能定義在類里面,它獨(dú)立于任何方法,不能定義在方法里面。
● 靜態(tài)代碼塊里面的變量都是局部變量,只在本塊內(nèi)有效。
● 靜態(tài)代碼塊會(huì)在類被加載時(shí)自動(dòng)執(zhí)行,而無論加載者是JVM還是其他的類。
● 一個(gè)類中允許定義多個(gè)靜態(tài)代碼塊,執(zhí)行的順序根據(jù)定義的順序進(jìn)行。
● 靜態(tài)代碼塊只能訪問類的靜態(tài)成員,而不允許訪問實(shí)例成員。
public class staticBlock{
//定義一個(gè)普通的main()方法
public static void main(String args[]){
System.out.println("This is main method.");
}
//定義一個(gè)靜態(tài)代碼塊
static{
System.out.println("This is static block.");
int stVar = 0; //這是一個(gè)局部變量,只在本塊內(nèi)有效
}
}
編譯通過后,用java命令加載本程序,會(huì)得到如下輸出:
This is static block.
This is main method.
從以上輸出結(jié)果中可以看出,靜態(tài)代碼塊甚至在main方法之前就被執(zhí)行。在main()方法中可以完成的任務(wù)在靜態(tài)代碼塊中都可以完成。但是二者在執(zhí)行上仍然有一些區(qū)別,main方法是整個(gè)程序啟動(dòng)的入口,而靜態(tài)代碼塊是存在于某個(gè)類中的一個(gè)過程。
5.靜態(tài)成員變量
Java允許以類作為靜態(tài)成員變量的類型,那么靜態(tài)成員變量就是一個(gè)對(duì)象。如果是基本數(shù)據(jù)類型的靜態(tài)成員變量,在類的外部可以不必創(chuàng)建對(duì)象就直接使用。但如果靜態(tài)成員是對(duì)象,問題就要復(fù)雜得多。因?yàn)閷?duì)象所屬的類,既可能有靜態(tài)成員,也可能有實(shí)例成員。而其中的實(shí)例成員必須要在對(duì)象實(shí)例化后才能使用,問題的核心在于:系統(tǒng)是否會(huì)為靜態(tài)的類變量創(chuàng)建實(shí)例 。
//-----------文件名supplyTest.java-----------------
public class supplyTest{
//定義一個(gè)靜態(tài)方法供測(cè)試用
public static void statShow(){
System.out.println("這是靜態(tài)方法");
}
//定義一個(gè)實(shí)例方法供測(cè)試用
public void instShow(){
System.out.println("這是實(shí)例方法");
}
}//end of supplyTest.java
//-----------文件名supplyTest.java-----------------
下面這個(gè)程序中,定義了一個(gè)supplyTest類型的變量,作為靜態(tài)成員,沒有顯示地實(shí)例化它。
//-----------文件名hasStatMember.java-----------------
public class hasStatMember{
static supplyTest stVar; //定義一個(gè)靜態(tài)成員
public static void main(String args[]){
stVar.statShow(); //調(diào)用靜態(tài)方法
stVar.instShow(); //調(diào)用實(shí)例方法
}
}
//-----------文件名hasStatMember.java-----------------
這個(gè)程序可以編譯通過,但它運(yùn)行的結(jié)果如下:
這是靜態(tài)方法
Exception in thread "main" java.lang.NullPointerException
at hasStatMember.main(hasStatMember.java:5)
從運(yùn)行結(jié)果中可以看出,靜態(tài)方法被正常執(zhí)行,但實(shí)例方法不能執(zhí)行,原因是未創(chuàng)建對(duì)象實(shí)例。這說明盡管stVar被聲明成static類型,系統(tǒng)仍然不會(huì)自動(dòng)為它創(chuàng)建對(duì)象,所以程序必須改成如下內(nèi)容才能正常運(yùn)行:
//-----------文件名hasStatMember.java-----------------
public class hasStatMember{
static supplyTest stVar = new supplyTest(); //定義一個(gè)靜態(tài)成員并實(shí)例化它
public static void main(String args[]){
stVar.statShow(); //調(diào)用靜態(tài)方法
stVar.instShow(); //調(diào)用實(shí)例方法
}
}
//-----------文件名hasStatMember.java-----------------
程序的輸出結(jié)果是:
這是靜態(tài)方法
這是實(shí)例方法
從輸出結(jié)果中可以看出,stVar的實(shí)例化是在定義時(shí)完成的,這意味著在hasStatMember類的外部可以像在內(nèi)部一樣使用它。下面這個(gè)程序演示了對(duì)stVar的使用形式。
//-----------文件名useStVar.java-----------------
public class useStVar{
public static void main(String args[]){
hasStatMember.stVar.statShow(); //調(diào)用靜態(tài)方法
hasStatMember.stVar.instShow(); //調(diào)用實(shí)例方法
}
}
//-----------文件名useStVar.java-----------------
程序的輸出結(jié)果如下:
這是靜態(tài)方法
這是實(shí)例方法
無論是靜態(tài)方法還是實(shí)例方法,都是通過“類名.靜態(tài)變量名.方法名”的形式來使用的。讀者可能會(huì)覺得這種形式有點(diǎn)眼熟。確實(shí)如此,前面大量使用的“System.out.println”就是這種形式。其中,System是系統(tǒng)預(yù)定義好的一個(gè)類,out是它的一個(gè)靜態(tài)成員,println是out的一個(gè)實(shí)例方法。
6.Java中的初始化順序
JAVA類首次裝入時(shí),會(huì)對(duì)靜態(tài)成員變量或方法進(jìn)行一次初始化,但方法不被調(diào)用是不會(huì)執(zhí)行的,靜態(tài)成員變量和靜態(tài)初始化塊級(jí)別相同,非靜態(tài)成員變量和非靜態(tài)初始化塊級(jí)別相同。
初始化順序:先初始化父類的靜態(tài)代碼--->初始化子類的靜態(tài)代碼-->
(創(chuàng)建實(shí)例時(shí),如果不創(chuàng)建實(shí)例,則后面的不執(zhí)行)初始化父類的非靜態(tài)代碼(變量定義等)--->初始化父類構(gòu)造函數(shù)--->初始化子類非靜態(tài)代碼(變量定義等)--->初始化子類構(gòu)造函數(shù)
類只有在使用New調(diào)用創(chuàng)建的時(shí)候才會(huì)被JAVA類裝載器裝入創(chuàng)建類實(shí)例時(shí),首先按照父子繼承關(guān)系進(jìn)行初始化類實(shí)例創(chuàng)建時(shí)候,首先初始化塊部分先執(zhí)行,然后是構(gòu)造方法;然后從本類繼承的子類的初始化塊執(zhí)行,最后是子類的構(gòu)造方法類消除時(shí)候,首先消除子類部分,再消除父類部分