public class A{
public static int item1=1;
public static final int item2 =2;
public void func(){
//do something
}
}
class B{
public static void main(String[] args{
A a = new A();
a.func();
System.out.println(a.item1+","+a.item2);
}
}
以上面代码为例,理顺一下Java代码编译得到class字节文件,到jvm中运行调度的过程。
1. 编译源文件到字节码,查找依赖的类。得到常量池(类型、成员变量)和方法字节码(方法引用、成员变量引用)。
2. 字节码文件的运行,首先找到main入口,将主类加载到运行的方法区(类的加载),执行main方法,加载初始化依赖A类的对象,马上Jvm加载A类,放在方法区。
3. 在堆区为A的实例分配内存,构造A实例,找到对应的方法引用
显然,上述代码运行结果是1,2。如果将A类中item1改成3,item2改成4,并且只重新编译A.java,运行B有什么结果呢,是3,2
另外注意:
1. Java中所有public,protected的实例方法都是动态绑定;所有私有方法、静态方法都是静态绑定。java中的变量都是静态绑定的,方法的话只有static和final(所有private默认是final的)是静态绑定的.
2. java向上转型(多态)就是采用了动态绑定。声明的是父类的引用,但是执行的过程中调用的是子类的对象,程序首先寻找子类对象的method方法,但是没有找到,于是向上转型去父类寻找。