請先看下面這段程序:
以下是代碼片段: public class Hello{ public static void main(String[] args){ //(1) System.out.println("Hello,world!"); //(2) } } |
看過這段程序,對于大多數(shù)學(xué)過Java 的從來說,都不陌生。即使沒有學(xué)過Java,而學(xué)過其它的高級語言,例如C,那你也應(yīng)該能看懂這段代碼的意思。它只是簡單的輸出“Hello,world”,一點(diǎn)別的用處都沒有,然而,它卻展示了static關(guān)鍵字的主要用法。
在1處,我們定義了一個(gè)靜態(tài)的方法名為main,這就意味著告訴Java編譯器,我這個(gè)方法不需要創(chuàng)建一個(gè)此類的對象即可使用。你還得你是怎么運(yùn)行這個(gè)程序嗎?一般,我們都是在命令行下,打入如下的命令(加下劃線為手動輸入):
javac Hello.java
java Hello
Hello,world!
這就是你運(yùn)行的過程,第一行用來編譯Hello.java這個(gè)文件,執(zhí)行完后,如果你查看當(dāng)前,會發(fā)現(xiàn)多了一個(gè)Hello.class文件,那就是第一行產(chǎn)生的Java二進(jìn)制字節(jié)碼。第二行就是執(zhí)行一個(gè)Java程序的最普遍做法。執(zhí)行結(jié)果如你所料。在2中,你可能會想,為什么要這樣才能輸出。好,我們來分解一下這條語句。(如果沒有安裝Java文檔,請到Sun的官方網(wǎng)站瀏覽J2SE API)首先,System是位于java.lang包中的一個(gè)核心類,如果你查看它的定義,你會發(fā)現(xiàn)有這樣一行:public static final PrintStream out;接著在進(jìn)一步,點(diǎn)擊PrintStream這個(gè)超鏈接,在METHOD頁面,你會看到大量定義的方法,查找println,會有這樣一行:
public void println(String x)。
好了,現(xiàn)在你應(yīng)該明白為什么我們要那樣調(diào)用了,out是System的一個(gè)靜態(tài)變量,所以可以直接使用,而out所屬的類有一個(gè)println方法。
靜態(tài)方法
通常,在一個(gè)類中定義一個(gè)方法為static,那就是說,無需本類的對象即可調(diào)用此方法。如下所示:
以下是代碼片段: class Simple{ static void go(){ System.out.println("Go..."); } } public class Cal{ public static void main(String[] args){ Simple.go(); } } |
調(diào)用一個(gè)靜態(tài)方法就是“類名.方法名”,靜態(tài)方法的使用很簡單如上所示。一般來說,靜態(tài)方法常常為應(yīng)用程序中的其它類提供一些實(shí)用工具所用,在Java的類庫中大量的靜態(tài)方法正是出于此目的而定義的。
靜態(tài)變量
靜態(tài)變量與靜態(tài)方法類似。所有此類實(shí)例共享此靜態(tài)變量,也就是說在類裝載時(shí),只分配一塊存儲空間,所有此類的對象都可以操控此塊存儲空間,當(dāng)然對于final則另當(dāng)別論了。看下面這段代碼:
以下是代碼片段: class Value{ static int c=0; static void inc(){ c++; } } class Count{ public static void prt(String s){ System.out.println(s); } public static void main(String[] args){ Value v1,v2; v1=new Value(); v2=new Value(); prt("v1.c="+v1.c+" v2.c="+v2.c); v1.inc(); prt("v1.c="+v1.c+" v2.c="+v2.c); } } |
結(jié)果如下:
v1.c=0 v2.c=0
v1.c=1 v2.c=1
由此可以證明它們共享一塊存儲區(qū)。static變量有點(diǎn)類似于C中的全局變量的概念。值得探討的是靜態(tài)變量的初始化問題。我們修改上面的程序:
以下是代碼片段: class Value{ static int c=0; Value(){ c=15; } Value(int i){ c=i; } static void inc(){ c++; } } class Count{ public static void prt(String s){ System.out.println(s); } Value v=new Value(10); static Value v1,v2; static{ prt("v1.c="+v1.c+" v2.c="+v2.c); v1=new Value(27); prt("v1.c="+v1.c+" v2.c="+v2.c); v2=new Value(15); prt("v1.c="+v1.c+" v2.c="+v2.c); } public static void main(String[] args){ Count ct=new Count(); prt("ct.c="+ct.v.c); prt("v1.c="+v1.c+" v2.c="+v2.c); v1.inc(); prt("v1.c="+v1.c+" v2.c="+v2.c); prt("ct.c="+ct.v.c); } } |
運(yùn)行結(jié)果如下:
v1.c=0 v2.c=0
v1.c=27 v2.c=27
v1.c=15 v2.c=15
ct.c=10
v1.c=10 v2.c=10
v1.c=11 v2.c=11
ct.c=11
這個(gè)程序展示了靜態(tài)初始化的各種特性。如果你初次接觸Java,結(jié)果可能令你吃驚。可能會對static后加大括號感到困惑。首先要告訴你的是,static定義的變量會優(yōu)先于任何其它非static變量,不論其出現(xiàn)的順序如何。正如在程序中所表現(xiàn)的,雖然v出現(xiàn)在v1和v2的前面,但是結(jié)果卻是v1和v2的初始化在v的前面。在static{后面跟著一段代碼,這是用來進(jìn)行顯式的靜態(tài)變量初始化,這段代碼只會初始化一次,且在類被第一次裝載時(shí)。如果你能讀懂并理解這段代碼,會幫助你對static關(guān)鍵字的認(rèn)識。在涉及到繼承的時(shí)候,會先初始化父類的static變量,然后是子類的,依次類推。非靜態(tài)變量不是本文的主題,在此不做詳細(xì)討論,請參考Think in Java中的講解。
靜態(tài)類
通常一個(gè)普通類不允許聲明為靜態(tài)的,只有一個(gè)內(nèi)部類才可以。這時(shí)這個(gè)聲明為靜態(tài)的內(nèi)部類可以直接作為一個(gè)普通類來使用,而不需實(shí)例一個(gè)外部類。如下代碼所示:
以下是代碼片段: public class StaticCls{ public static void main(String[] args){ OuterCls.InnerCls oi=new OuterCls.InnerCls(); } } class OuterCls{ public static class InnerCls{ InnerCls(){ System.out.println("InnerCls"); } } } |
輸出結(jié)果會如你所料:
InnerCls
和普通類一樣。內(nèi)部類的其它用法請參閱Think in Java中的相關(guān)章節(jié),此處不作詳解。
static指向同一塊內(nèi)存,當(dāng)你把某數(shù)據(jù)成員或函數(shù)聲明為static時(shí),它就不再局限于所屬的class object,這樣就有了一個(gè)好處,可以讓你在不建立任何對象的情況下,調(diào)用你聲明的函數(shù)。
舉一個(gè)例子:
class Test{
static int i=1;
}
Test t1=new Test();
Test t2=new Test();
那么t1.i和t2.i都是1,它們其中任一個(gè)值改變,另外的也會一起改變。