所有main方法都被标记为static修饰符。下面讨论这个修饰符的含义
静态域
如果将域定义为 static, 每个类中只有一个这样的域。而每一个对象对于所有的实例域却都有自己的一份拷贝。。例如, 假定需要给每一个雇员賦予唯一的标识码。这里给 Employee类添加一个实例域 id 和一个静态域 nextld:
class Employee
{
private static int nextld = 1;
private int id;
}
现在, 每一个雇员对象都有一个自己的 id 域, 但这个类的所有实例将共享一个 nextld
域。换句话说, 如果有 1000 个 Employee 类的对象, 则有 1000 个实例域 id。但是, 只有一
个静态域 nextld。即使没有一个雇员对象, 静态域 nextld 也存在。它属于类,而不属于任何
独立的对象。
静态常量
静态变量使用得比较少,但静态常量却使用得比较多。例如, 在 Math 类中定义了一个静态常量:
public class Math
{
public static final double PI = 3.14159265358979323846;
}
在程序中,可以采用 Math.PI 的形式获得这个常量。如果关键字 static 被省略, PI 就变成了 Math 类的一个实例域。需要通过 Math 类的对象访问 PI,并且每一个 Math 对象都有它自己的一份 PI 拷贝。
另一个多次使用的静态常量是 System.out。它在 System 类中声明:
public class System
{
...
public static final PrintStream out = . . .;
...
}
前面曾经提到过,由于每个类对象都可以对公有域进行修改,所以,最好不要将域设计为 public。然而,== 公有常量(即 final 域)却没问题==。因为 out 被声明为 final, 所以,不允许再将其他打印流赋给它:
System.out = new PrintStrean(. . .); // Error out is final
静态方法
静态方法是一种不能向对象实施操作的方法。例如, Math 类的 pow 方法就是一个静态方法。表达式
Math.pow(x, a)
计算x的a次幂。在运算时, 不使用任何 Math 对象。换句话说,没有隐式的参数。可以认为静态方法是没有 this 参数的方法(在一个非静态的方法中,this 参数表示这个方法的隐式参数)。
静态方法可以访问自身类中的静态域。下面是使用这种静态方法的一个示例:
public static int getNextld()
{
return nextld; // returns static field
}
可以通过类名调用这个方法:
int n = Employee.getNextld();
在下面两种情况下使用静态方法:
- 一个方法不需要访问对象状态,其所需参数都是通过显式参数提供(例如:Math.pow)
- 一个方法只需要访问类的静态域(例如:Employee.getNextld)
工厂方法
静态方法还有另外一种常见的用途。类似 LocalDate 和 NumberFormat 的类使用静态工厂方法 (factory method)来构造对象。你已经见过工厂方法 LocalDate.now 和 LocalDate.of。NumberFormat 类如下使用工厂方法生成不同风格的格式化对象:
NumberFormat currencyFormatter = NumberFormat.getCurrencyInstance();
NumberFormat percentFormatter = NumberFormat.getPercentInstance();
double x = 0.1;
System.out.println(currencyFormatter.format(x)); // prints O.10
System.out.println(percentFomatter.format(x)); // prints 10%
为什么 NumberFormat 类不利用构造器完成这些操作呢? 这主要有两个原因:
-
无法命名构造器。构造器的名字必须与类名相同。但是, 这里希望将得到的货币实例 和百分比实例采用不用的名字。
-
当使用构造器时,无法改变所构造的对象类型。而 Factory 方法将返回一个 DecimalFormat 类对象,这是 NumberFormat 的子类。
main方法
需要注意,不需要使用对象调用静态方法。例如,不需要构造 Math 类对象就可以调用Math.pow。同理, main 方法也是一个静态方法。
public class Application
{
public static void main(StringD args)
{
// construct objects here
}
}
main 方法不对任何对象进行操作。事实上,在启动程序时还没有任何一个对象。静态的main 方法将执行并创建程序所需要的对象。