靜態方法

 

3.7  靜 態 方 法

前面已經介紹過,成員變量分爲實例變量和靜態變量。其中實例變量屬於某一個具體的實例,必須在類實例化後才真正存在,不同的對象擁有不同的實例變量。而靜態變量被該類所有的對象公有(相當於全局變量),不需要實例化就已經存在。

方法也可分爲實例方法和靜態方法。其中,實例方法必須在類實例化之後通過對象來調用,而靜態方法可以在類實例化之前就使用。與成員變量不同的是:無論哪種方法,在內存中只有一份——無論該類有多少個實例,都共用同一個方法。

本節以前的例子中,除了main()方法,其餘的方法都是實例方法,而main()則是一個靜態方法,所以它才能夠被系統直接調用。

3.7.1  靜態方法的聲明和定義

定義一個靜態方法和定義一個實例方法,在形式上並沒有什麼區別,只是在聲明的頭部,需要加上一個關鍵字static。它的一般語法形式如下:

[訪問權限修飾符] static [返回值類型] 方法名([參數列表]){

   語句序列

}

例如下面是一個靜態的方法:

public  static  void stFun(){

     System.out.println("這是一個靜態方法");

}

3.7.2  靜態方法和實例方法的區別

靜態方法和實例方法的區別主要體現在兩個方面:

  ●     在外部調用靜態方法時,可以使用類名.方法名的方式,也可以使用對象名.方法名的方式。而實例方法只有後面這種方式。也就是說,調用靜態方法可以無需創建對象。

  ●     靜態方法在訪問本類的成員時,只允許訪問靜態成員(即靜態成員變量和靜態方法),而不允許訪問實例成員變量和實例方法;實例方法則無此限制。

下面幾個例子展示了這一區別。

【例3.23】 調用靜態方法示例。

//-----------文件名hasStaticMethod.java,程序編號3.35-----------------

public class hasStaticMethod{

   //定義一個靜態方法

   public static void callMe(){

    System.out.println("This is a static method.");

  }

}

下面這個程序使用兩種形式來調用靜態方法。

//-----------文件名invokeStaticMethod.java,程序編號3.36-----------------

public class invokeStaticMethod{

  public static void main(String args[]){

    hasStaticMethod.callMe();        //不創建對象,直接調用靜態方法   

    hasStaticMethod oa = new hasStaticMethod();   //創建一個對象

    oa.callMe();                     //利用對象來調用靜態方法

  }

}

程序3.36兩次調用靜態方法,都是允許的,程序的輸出如下:

This is a static method.

This is a static method.

允許不創建對象而調用靜態方法,是Java爲了減少程序員調用某些常用方法時的麻煩,而允許程序員按照傳統的C語言中使用函數的方式來使用方法。典型的例子是前面某些程序中使用“Math.ramdon()”來獲取隨機數。

【例3.24】 靜態方法訪問成員變量示例。

//-----------文件名accessMember.java,程序編號3.37-----------------

class accessMember{

  private static int sa; //定義一個靜態成員變量

  private         int ia;     //定義一個實例成員變量

  //下面定義一個靜態方法

   static void statMethod(){

     int i = 0;              //正確,可以有自己的局部變量

         sa = 10;        //正確,靜態方法可以使用靜態變量

         otherStat();    //正確,可以調用靜態方法

         ia = 20;        //錯誤,不能使用實例變量

         insMethod();    //錯誤,不能調用實例方法

    }

   static void otherStat(){

   }

   //下面定義一個實例方法

   void  insMethod(){

     int i = 0;              //正確,可以有自己的局部變量

       sa = 15;              //正確,可以使用靜態變量

       ia = 30;              //正確,可以使用實例變量

      statMethod();          //正確,可以調用靜態方法

  }

}

本例其實可以概括成一句話:靜態方法只能訪問靜態成員,實例方法可以訪問靜態和實例成員。之所以不允許靜態方法訪問實例成員變量,是因爲實例成員變量是屬於某個對象的,而靜態方法在執行時,並不一定存在對象。同樣,因爲實例方法可以訪問實例成員變量,如果允許靜態方法調用實例方法,將間接地允許它使用實例成員變量,所以它也不能調用實例方法。基於同樣的道理,靜態方法中也不能使用關鍵字this

main()方法是一個典型的靜態方法,它同樣遵循一般靜態方法的規則,所以它可以由系統在創建對象之前就調用。下面這個程序有個錯誤,請讀者仔細查看。

public class hasError{

  int insVar = 100;

  public static void main(String args[]){

    System.out.println("insVar = " + insVar);

  }

}

3.7.3  靜態代碼塊

在類中,可以將某一塊代碼聲明爲靜態的,這樣的程序塊叫靜態初始化段。靜態代碼塊的一般形式如下:

static {

  語句序列

}

  ●     靜態代碼塊只能定義在類裏面,它獨立於任何方法,不能定義在方法裏面。

  ●     靜態代碼塊裏面的變量都是局部變量,只在本塊內有效。

  ●     靜態代碼塊會在類被加載時自動執行,而無論加載者是JVM還是其他的類。

  ●     一個類中允許定義多個靜態代碼塊,執行的順序根據定義的順序進行。

  ●     靜態代碼塊只能訪問類的靜態成員,而不允許訪問實例成員。

【例3.25】 靜態代碼塊運行示例1

//-----------文件名staticBlock.java,程序編號3.38-----------------

public class staticBlock{

  //定義一個普通的main()方法

  public static void main(String args[]){

    System.out.println("This is main method."); 

  }

  //定義一個靜態代碼塊

  static{

     System.out.println("This is static block.");

     int stVar = 0;   //這是一個局部變量,只在本塊內有效

  }

}

編譯通過後,用java命令加載本程序,會得到如下輸出:

This is static block.

This is main method.

從以上輸出結果中可以看出,靜態代碼塊甚至在main方法之前就被執行。在main()方法中可以完成的任務在靜態代碼塊中都可以完成。但是二者在執行上仍然有一些區別,請看下例。

【例3.26】 靜態代碼塊和main()方法的區別。

這裏仍然使用例3.25中的staticBlock類,然後新定義一個類來使用它。

//-----------文件名useStaticBlock.java,程序編號3.39-----------------

public class useStaticBolck{

  public static void main(String args[]){

     new staticBlock();  //創建一個staticBlock的對象

  }

}

本程序沒有像以前的程序那樣,在創建對象時使用一個變量來接收對象,因爲這個程序在後面並不需要用到這個變量。程序的輸出如下:

This is static block.

這一次,只執行了靜態代碼塊,main()方法在這種情況下是不會被執行的。

最後來寫一個複雜一點的靜態代碼塊的例子,它綜合體現了靜態代碼塊的使用方法,請讀者注意註釋說明。

【例3.27】 靜態代碼塊使用示例2

//-----------文件名staticBlock.java,程序編號3.40-----------------

public class staticBlock{

  static int stMember = 100;            //定義靜態成員變量

  public static void main(String args[]){

    System.out.println("This is main method.");

  }

  //第一個靜態代碼塊

  static{

     System.out.println("This is first static block.");

     stMember  = 200;                        //訪問靜態成員變量

     staticBlock oa = new staticBlock(); //創建對象

     System.out.println("stMember = " + oa.stMember);

     statFun();                              //調用靜態方法

  }

  //定義一個靜態方法

  static void statFun(){

     System.out.println("This is a static method.");

  }

  //第二個靜態代碼塊

  static{

     System.out.println("This is second static block.");

  }

}

程序運行的結果如下:

This is first static block.

stMember = 200

This is a static method.

This is second static block.

This is main method.

3.7.4  再論靜態成員變量

在前面的3.3.3節中已經介紹過靜態成員變量,不過那裏的靜態成員變量都是一些基本類型。Java允許以類作爲靜態成員變量的類型,那麼靜態成員變量就是一個對象。

如果是基本數據類型的靜態成員變量,在類的外部可以不必創建對象就直接使用。但如果靜態成員是對象,問題就要複雜得多。因爲對象所屬的類,既可能有靜態成員,也可能有實例成員。而其中的實例成員必須要在對象實例化後才能使用,問題的核心在於:系統是否會爲靜態的類變量創建實例。

【例3.28】 對象作爲靜態成員使用示例。

//-----------文件名supplyTest.java,程序編號3.41-----------------

public class supplyTest{

  //定義一個靜態方法供測試用

  public static void statShow(){

     System.out.println("這是靜態方法");

  }

  //定義一個實例方法供測試用

  public void  instShow(){

      System.out.println("這是實例方法");

  }

}

下面這個程序中,定義了一個supplyTest類型的變量,作爲靜態成員,沒有顯示地實例化它。

//-----------文件名hasStatMember.java,程序編號3.42-----------------

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類型,系統仍然不會自動爲它創建對象,所以程序3.42必須改成如下內容才能正常運行:

//-----------文件名hasStatMember.java,程序編號3.42-----------------

public class hasStatMember{

  static supplyTest stVar = new supplyTest();     //定義一個靜態成員並實例化它

  public static void main(String args[]){

     stVar.statShow();                             //調用靜態方法

     stVar.instShow();                             //調用實例方法

  }

}

程序的輸出結果是:

這是靜態方法

這是實例方法

從輸出結果中可以看出,stVar的實例化是在定義時完成的,這意味着在hasStatMember類的外部可以像在內部一樣使用它。下面這個程序演示了對stVar的使用形式。

//-----------文件名useStVar.java,程序編號3.43-----------------

public class useStVar{

  public static void main(String args[]){   

     hasStatMember.stVar.statShow();     //調用靜態方法

     hasStatMember.stVar.instShow();     //調用實例方法

  }

}

程序的輸出結果如下:

這是靜態方法

這是實例方法

無論是靜態方法還是實例方法,都是通過類名.靜態變量名.方法名的形式來使用的。讀者可能會覺得這種形式有點眼熟。確實如此,前面大量使用的“System.out.println”就是這種形式。其中,System是系統預定義好的一個類,out是它的一個靜態成員,printlnout的一個實例方法。

 

 

 

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章