如果將域定義為static,那麼每個類中只有一個這樣的域。作為對比,每一個對象對於所有的執行個體域卻都有自己的一份拷貝。
例如,假定需要給每一個僱員賦予唯一的標識碼。這裡給Employee類添加一個執行個體域id和一個靜態域nextId:
class Employee{ ...... private int id; private static int nextId = 1;}
現在,每一個僱員對象都有一個自己的id域,但這個類的所有執行個體將共用一個nextId域。換句話說,如果有1000個Employee類的對象,
則有1000個執行個體域id,其中任一執行個體使用了setId()方法後,Employee類的靜態域都會變成設定後的值。但是,只有一個靜態域nextId。
即使沒有一個僱員對象,靜態域nextId也存在。它屬於類,而不屬於任何獨立的對象。從靜態域通過類名直接調用,執行個體域通過(類的)執行個體調用,也可佐證。
項目中遇到的一個例子:如果寫一個類XDialog繼承Dialog,XDialog裡面有個域變數private List<String> items;我每次要顯示一個對話方塊時都new XDialog();這時這個對話方塊就有了一個items拷貝,new N個XDialog就有N個對應的items拷貝。但若items是指目前使用者的好友數目,那麼XDialog對話方塊中就應該時時顯示正確的上/下線好友,若是N個items拷貝,就每次都初始化了items,不能達到同步的目的,且效率低。這時用static關鍵字就解決了所有問題,只需聲明privatestatic List<String> items;這時items為XDialog這個類所有,而非某個實現。Perfect!
再來看下靜態方法:
靜態方法是不能向對象實施操作的方法。
例如,Math.pow(x, a),它在運算的時候,不使用任何Math對象。換句話說,沒有隱式的參數(this)。
可以認為靜態方法是沒有this參數的方法。(在一個非靜態方法中,this參數表示該方法的隱式參數。)作為對比:非靜態方法的調用(this.)pow();
因為靜態方法不能操作對象,所以不能在靜態方法中訪問執行個體域。但是,靜態方法可以訪問自身類中的靜態域。
public static int getNextId(){ return nextId;//return static field} 通過類名調用這個方法:int n = Employee.getNextId();
註:靜態域不能直接存取執行個體域(因為它是與類相關的而不是某個對象),但執行個體域可直接存取靜態域(因為執行個體域是派生於類的)。
如果harry是一個Employee對象,可用harry.getNextId()來替代Employee.getNextId()。不過,這種方式很容易讓人迷惑。因為getNextId方法計算的結果與harry毫無關係。故建議使用類名,而不是對象來調用靜態方法。
在下面兩種情況下使用靜態方法:
1.當一個方法不需要訪問對象狀態,其所需參數都是通過顯式提供的(例如Math.pow)。
2.當一個方法只需要訪問類的靜態域(例如Employee.getNextId)。
接下來看下靜態內部類:
有時候,使用內部類只是為了把一個類隱藏在另外一個類的內部,並不需要內部類引用外圍類對象。為此,可以將內部類聲明為static,以便取消產生的引用。
靜態內部類的對象除了沒有對產生它的外圍類對象的引用特權外,與其他所有內部類完全一樣。
例如,定義靜態內部類Pair:
class ArrayAlg{ public static class Pair{//不需要內部類引用外圍對象,用static斷絕這種引用。 private double first; private double second; public Pair(double f, double s){ first = f; second = s; } public double getFirst(){ return first; } public double getSecond(){ return second; } public static Pair minmax(double[] values){ double min = Double.MAX_VALUE; double max = Double.MIN_VALUE; for(double v:values){ if(min>v) min = v; if(max<v) max = v; } return new Pair(min,max); } }//end Pair}
其中,ArrayAlg是內部類,Pair是靜態內部類。
在主main中:
ArrayAlg.Pair p = ArrayAlg.Pair.minmax(d);//d:double數組
System.out.println("min="+p.getFirst()+",max="+p.getSecond());