class Father{
int i=10;
public void printInfo(){
System.out.println("In Father");
}
}class Son extends Father{
int i=20;
public void printInfo(){
System.out.println("In Son");
}
}class Test{
public static void main(String args[]){
Son mySon=new Son(); //调用子类的变量和子类的方法
Father myFather=new Father(); //调用父类的变量和方法的方法
Father myFather2=mySon; //调用父类的变量和子类的方法
//mySon
System.out.println(mySon.i);
mySon.printInfo();
System.out.println();
//myFather
System.out.println(myFather.i);
myFather.printInfo();
System.out.println();
//myFather2
System.out.println(myFather2.i);
myFather2.printInfo();
System.out.println();
}
}
int i=10;
public void printInfo(){
System.out.println("In Father");
}
}class Son extends Father{
int i=20;
public void printInfo(){
System.out.println("In Son");
}
}class Test{
public static void main(String args[]){
Son mySon=new Son(); //调用子类的变量和子类的方法
Father myFather=new Father(); //调用父类的变量和方法的方法
Father myFather2=mySon; //调用父类的变量和子类的方法
//mySon
System.out.println(mySon.i);
mySon.printInfo();
System.out.println();
//myFather
System.out.println(myFather.i);
myFather.printInfo();
System.out.println();
//myFather2
System.out.println(myFather2.i);
myFather2.printInfo();
System.out.println();
}
}
当一个父类定义的变量引用一个子类实例时,调用一个方法时,这个方法将会调用子类,因为方法被覆盖.情况就特殊在父类定义变量,而引用子类对像的时候.
Father(父类) myFather = new Son(子类)();
通过引用类型的变量来访问所引用对象的方法和属性的时候,jvm采用以下绑定规则。
实例方法与引用变量实际引用的对象的方法绑定,动态绑定,因为是在运行时由jvm动态决定的
静态方法,成员变量(静态和实例变量)与引用变量所声明的静态方法和成员变量绑定,这用绑定是在编译阶段做的绑定,是静态绑定。但是java编译器编译的时候都是按照声明的引用变量来处理的。
例如
在Father类中如果没有i=10的话,
Son类有i=20
System.out.println(myFather2.i);
上面这句话在编译的时候就报错,编译都是无法通过的。
在面向对象中,可以吧函数调用理解为“消息传递”。Father f = new Father();
f.method(); //可以理解为向对象f发送method消息,实际的调用是 method(f),
//括号里是放松消息的目标对象
f = new Son();
f.method(); //method(f),这里f对象实际是Son,所以是把消息发送给Son对象,自然调用的子类方法这样似乎比较好理解继承中的方法调用问题。