java 父类 子类 引用 默认构造函数

来源:百度知道 编辑:UC知道 时间:2024/07/03 01:22:27
代码如下,运行结果是0
3
0
3
3
0

请问为什么是这个结果,过程是什么?
X s = new Y();//为什么通过Y()默认构造函数创建s时候,子类的i,j没有覆盖掉父类的i,j?

class X {
int i = 0;

int j = 3;

// X(int a,int b){
// System.out.println(a+",,,,,"+b);
// }

// void out() {
// System.out.println(i + " " + j);
// }
}

class Y extends X {
int i = 3;

int j = 0;
// Y(int a,int b){
// super(a,b);
// System.out.println(a+","+b);
// }

void out() {
System.out.println(i + " " + j);
}
}

public class MianShiTi {

public static void main(String[] args) {
X s = new Y();
System.out.println(s.i);
System.out.println(s.j);
X aa = new X();
System.out.println(aa.i);
System.out.println(aa.j);
Y bb = new Y();
System.out.println(bb.i);
System.out.println(bb

我想LZ之所以对java的这种行为感到奇怪,是因为你把类里的属性和方法来做了对比吧。也就是子类(Y)同时覆盖了父类(X)的属性和方法时,为什么“X s = new Y(); ”这里的s只可以调用有Y中覆盖X的方法,而不能使用Y中覆盖X的属性。这也是疑惑的关键。

我先给出一个结论:父类引用指向子类对象时,父类与子类同名的属性和方法都会被覆盖。 (一般对属性的覆盖叫对属性的隐藏)

对于上面的结论我们来看你这个程序,既然属性可以被覆盖,那怎么s.i是用了父类的变量值呢?这是关系到了java处理属性和方法的不同机制引起的。

先再给出一个结论:java对方法的调用是采取动态绑定(运行时才判断)的方法,对属性的调用是采取静态绑定(编译时就判断)的方法。

对于上面的结论我们来理解你的程序“X s = new Y(); ”,这一句,也可说是父类引用指向子类对象,也可以说是子类对象向上转型为父类引用。那在s使用属性的时候是静态绑定的,它是在编译期就决定了这个属性是属于哪一个类的对象了,所以在上面那句中X s.也就是说X类中的属性是用s来调用的,s就不能调用Y类中的属性,因为它是静态绑定了。而s使用方法时是动态绑定的,所以s在调用方法的时候,java会去查看s这个时候的指向的对象是哪一个,是Y类对象就调用Y类的方法,是X类的对象就调用X类的方法。所以这里s是指向了Y类中的对象(“X s = new Y(); ”这句可知),则调用Y类中的方法。

好,上面对你的问题讲清了,下面再说说java因为什么对属性要采取静态的绑定方法。这是因为静态绑定是有很多的好处,它可以让我们在编译期就发现程序中的错误,而不是在运行期。这样就可以提高程序的运行效率!而对方法采取动态绑定是为了实现多态,知道java是的多态是java的一大特色。多态也是面向对象的关键技术之一,所以java是以效率为代价来实现多态这是很值得的

父类引用指向子类对象,不能调用子类新定义的变量,方法,即使是同名也不会被覆盖.

楼上说的不完全对;
父类引用指向子类对象,父类的同名变量不会被覆盖,但父类的同名方法会被覆盖。

麻烦你去把多态搞清楚。
java最基本的