总结:
java的动态绑定机制:成员方法在执行的过程中
- JVM会将当前方法和当前调用对象实际内存进行绑定,
- 但是成员属性是没有动态绑定机制的,属性在哪里声明就在哪里使用。
结论的来源理论依据来自于jvm编译原理,有兴趣的可以亲自去试试。
以下通过一个简单的继承关系来说明
class Test{
public static void main(String args[]){
AAA aaa=new BBB();
System.out.println(aaa.getRes());
}
}
class AAA{
public int i=20;
public int getRes()
{
return i+10;
}
}
class BBB extends AAA{
public int i=10;
public int getRes()
{
return i+10;
}
}
这里打印的结果是 20
分析,当调用aaa.getRes()的时候,会先去当前对象对应的内存,子类和父类中都存在这个方法,因此调用的是子类B中的方法,方法找到了,但是用的是哪个i呢,我们通过断点调试,可以看到aaa里面是有两个i的,具体使用的是哪个呢,我们可以看到有一个i是标记了的AAA的,对于成员变量使用的是当前对象对应的内存中的值,也就是10。
变化1
class Test{
public static void main(String args[]){
AAA aaa=new BBB();
System.out.println(aaa.getRes());
}
}
class AAA{
public int i=20;
public int getRes()
{
return i+10;
}
}
class BBB extends AAA{
public int i=10;
// public int getRes()
// {
// return i+10;
// }
}
本次执行结果是30,
这里子类没有调用的方法,就去调用父类的,但是父类里面有一个i,那么这里的i是调用哪个呢?通过上面的结论,在哪里声明就在哪里调用,也就是20+10=30
变化2
这里打印结果是20,很明显这次获取到的i就是10了,这是为啥呢,通过上面的结论我们可以知道,成员方法是动态绑定机制的,因此获取到的是当前对象也就是B里面的10,调用的方法是A中的因此是10+10是20。