先看一段代码:
import java.util.HashMap;
import java.util.Map;
class Father{
String name = "father";
public String getName(){
return name;
}
}
class Son extends Father{
String name = "son";
@Override
public String getName(){
return name;
}
}
public class Test {
public static void main(String[] args) throws InterruptedException {
Father son = new Son();
System.out.println(son.name); //father
System.out.println(son.getName()); //son
}
}
为什么通过son.getName()可以获取到想要的内容?
因为这里涉及到了方法重写,在虚拟机执行过程中会通过一个invokeVritual的指令来确定到底执行哪个方法。该指令会通过运行时方法接收者的实际类型来确定执行哪个方法,其解析过程如下:
① 找到操作数栈顶第一个元素(接收者)的实际类型,记为C。
② 看C中是否存在目标方法,如果存在则进行权限校验,检验通过则返回该方法直接引用,校验未通过则抛出java.lang.IllegalAccessError异常。
③ 如果C中不存在目标方法,则向上对其父类进行相同查找。
④ 父类中都没有找到合适的方法,则抛出java.lang.AbstractMethodError异常。
在这个例子中,接收者实际类型是Son,而其中就有一个简单名称和描述符都一样的方法,那就确定最终调用的就是Son::getName()方法;而Son::getName()返回的是子类中的name。
为什么通过son.name却不行?
因为Java中字段是不参与多态的。在这个例子中,son对象的外观类型是Father,因此调用son.name得到的就是Father类中的name属性。
遇到这种需求时如何解决?
尽量避免通过对象直接访问其内部属性,可以为属性设置get\set方法来访问其内部属性。