static 表示静态、全局的意思。
用处:
1 静态域
2 静态常量
3 静态方法
4 静态代码块
(一)、静态域
如果将域定义为static,每个类中只有一个这样的域。而每一个对象对于所有的实例域却都有自己的一份拷贝。例如,这里给Employee类添加一个实例域id和一个静态域nextId:
class Employee{
private static int nextId=1;
private int id;
. . .
}
现在,每一个Employee对象都有一个自己的id域,但这个类的所有实例将共享一个nextId域。换句话说,如果有100个Employee类的对象,则有100个实例域id。但是,只有一个静态域nextId。即使没有一个Employee对象,静态域nextId也存在。它属于类,而不属于任何独立的对象。
在绝大数的面向对象程序设计语言中,静态域被称为类域。
可以直接通过类名调用(Employee.nextId),也可以通过该类的对象调用(new Employee().nextId)。
(二)、静态常量
静态变量使用的比较少,但静态常量却使用得比较多。例如,在Math类中定义了一个静态常量:
publi class Math{
public static final double PI=3.1415926;
}
在程序中,可以采用Math.PI的形式获得这个常量。
也可以通过Math类的对象访问PI。(不推荐这样使用)
另一个多次使用的静态常量是System.out。它在System类中声明:
public class System{
....
public static fianl PrintStream out =...;
}
前面曾经提到过,由于每个类对象都可以对公有域进行修改,所以,最好不要将域设计为public。然而,公有常量(即final域)却没有问题。因为out被声明为final,所以,不允许再将其他打印流赋给它:
System.out=new PrintStream(); //Error
注:如果查看一下System类中,就会发现有一个setOut方法,它可以将System.out设置为不同的流。为什么这个方法可以修改final变量的值。原因在于,setOut方法是一个本地方法,而不是用java语言实现的。本地方法可以绕过java语言的存取控制机制。
(三)、静态方法
静态方法是一种不能向对象实施操作的方法。例如在Math类的pow方法就是一个静态方法。表达式
Math.pow(x,a)计算幂x^a。在运算时,不适用任何Math对象。
可以认为静态方法是没有this的方法,在static方法内部不能调用非静态方法,反过来是可以的。Employee类的静态方法不能访问id实例域,因为它不能操作对象。但是,静态方法可以访问自身类中的静态域。下面是使用静态方法的一个示例:
public static int getNextId(){
return nextId;//return static field
}
可以通过类名调用这个方法:int n=Employee.getNextId(); 可以省略关键字static。
可以通过对象调用静态方法。不过这种方式容易造成混淆,不建议使用类名。
使用静态方法的情况:
1 一个方法不需要访问对象状态,其所需参数都是通过显示参数提供(如:Math.pow)。
2 一个方法内部只需要访问类的静态域(如Employee.getNextId).
(四)、静态代码块
静态块用于初始化类,为类的属性初始化。每个静态代码块只会执行一次。由于JVM在加载类时会执行静态代码块,所以静态代码块先于主方法执行。
如果类中包含多个静态代码块,那么将按照"先定义的代码先执行,后定义的代码后执行"。
注意:1 静态代码块不能存在于任何方法体内。2 静态代码块不能直接访问静态实例变量和实例方法,需要通过类的实例对象来访问。
class Code{
{
System.out.println("Code的构造块");
}
static{
System.out.println("Code的静态代码块");
}
public Code(){
System.out.println("Code的构造方法");
}
}
public class CodeBlock03{
{
System.out.println("CodeBlock03的构造块");
}
static{
System.out.println("CodeBlock03的静态代码块");
}
public CodeBlock03(){
System.out.println("CodeBlock03的构造方法");
}
public static void main(String[] args){
System.out.println("CodeBlock03的主方法");
new Code();
new Code();
new CodeBlock03();
new CodeBlock03();
}
}
输出:
CodeBlock03的静态代码块
CodeBlock03的主方法
Code的静态代码块
Code的构造块
Code的构造方法
Code的构造块
Code的构造方法
CodeBlock03的构造块
CodeBlock03的构造方法
CodeBlock03的构造块
CodeBlock03的构造方法