Java静态字段(属性、方法、类别)
假设域被定义为static,那么每个类中仅仅有一个这种域。作为对照,每个对象对于全部的实例域却都有自己的一份拷贝。
比如,假定须要给每个雇员赋予唯一的标识码。
这里给Employee类加入一个实例域id和一个静态域nextId:
class Employee{
......
private int id;
private static int nextId = 1;
}
如今,每个雇员对象都有一个自己的id域,但这个类的全部实例将共享一个nextId域。
换句话说,假设有1000个Employee类的对象。
则有1000个实例域id,当中任一实例使用了setId()方法后。Employee类的静态域都会变成设置后的值。可是,仅仅有一个静态域nextId。
即使没有一个雇员对象。静态域nextId也存在。它属于类,而不属于不论什么独立的对象。从静态域通过类名直接调用,实例域通过(类的)实例调用,也可佐证。
静态方法
静态方法是不能向对象实施操作的方法。
比如,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()。只是,这样的方式非常easy让人迷惑。由于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());