在进入正式讨论之前我们先介绍一下什么是实例变量和类变量?
通俗来说用static 修饰的变量是类变量,其余变量可以称为实例变量。同样使用static 修饰的方法称作类方法,不用static 修饰的方法称作实例方法。
实例方法可以调用该类中的实例方法的类方法,而类方法只能调用该类的类方法。
1.两者内存情况分析:
当JAVA程序执行时,类的字节码文件被加载到内存上,如果该类没有创建内存,类中的实例变量就不会分配内存。但是,类中的类变量在该类被加载到内存时,就分配了相应的内存空间。如果该类创建对象,那么不同对象的实例变量就互相不相同,即分配不同的空间,而类方法不在重新分配内存,所有的对象共享类变量,即所有对象的类变量是相同的一处内存空间,类变量的内存空间直到程序退出运行才释放所有内存。
2.访问权限:
改变其中一个对象的类变量就同时改变了其他对象的这个类变量,因此,类遍历那个不仅可以通过某个对象访问,也可直接通过类名访问。实例变量只能通过对象访问。
例:
package chapter04;
public class Lader {
double 上底,高;
static double 下底;
void 设置上底(double a){
上底 = a;
}
void 设置下底(double b){
下底 = b;
}
double 获取上底(){
return 上底;
}
double 获取下底(){
return 下底;
}
}
package chapter04;
public class Example4_8 {
public static void main(String args[]){
Lader.下底 = 100;
Lader laderOne = new Lader();
Lader laderTwo = new Lader();
laderOne.设置上底(28);
laderTwo.设置上底(66);
System.out.println("laderOne的上底:"+laderOne.获取上底());
System.out.println("laderOne的下底:"+laderOne.获取下底());
System.out.println("laderTwo的上底:"+laderTwo.获取上底());
System.out.println("laderTwo的下底:"+laderTwo.获取下底());
}
}
运行结果:
laderOne的上底:28.0
laderOne的下底:100.0
laderTwo的上底:66.0
laderTwo的下底:100.0
内存示意图
问题来了:为什么类方法只能调用类变量和类中其余类方法呢?
其实当类的字节码文件被加载到内存时,类的实例方法不会分配入口地址。在该类创建对象后,类中的实例方法才分配到了入口地址,从而实例方法可以被类创建的任何对象调用执行,需要注意,创建第一个对象时类中实例方法就分配了入口地址,当所有对象都不存在时,方法的入口地址才被取消。
对于类中的类方法,在该类被加载到内存时,就分配了相应的入口地址,从而类方法不仅可以可以被类创建的任何对象调用执行,也可直接通过类名调用执行。类方法的入口地址到程序退出时才被取消。