黑马程序员技术交流社区
标题: 关于方法的覆写问题 [打印本页]
作者: 创出一片辉煌 时间: 2012-7-27 01:33
标题: 关于方法的覆写问题
继承, 覆写, private, default
class Person{ //定义父类
private void print(){
System.out.println("Person");
}
public void fun(){
this.print(); //调用父类中的print方法
}
}
class Student extends Person{ //定义子类
void print(){
System.out.println("Student");
}
}
class OverrideDemo04{
public static void main(String args[]){
Student s = new Student();
s.fun();
s.print();
}
}
复制代码以上代码,执行的结果如下:
Person
Student
可见Student类中的print方法成功覆写,然后调用继承自Person类的fun方法时,能成功访问到Person类中的private方法。
想请问:
Student类的实例对象s,为何能通过fun方法访问到父类中的private方法?
当s调用fun方法的时候,fun方法的定义里面的this关键字不是应该等同于s么?
作者: 王志明 时间: 2012-7-27 01:45
本帖最后由 王志明 于 2012-7-27 02:39 编辑
楼主你理解错了,不要被运行结果的表象骗到
1. 在你的Student类中并没有覆盖父类中的print方法,因为父类中的print方法是private的,你并没有继承,不能直接访问,你在Student类中写的print方法是Student特有的,所有当你s.print()是,输出--->Student
2.当s调用fun方法的时候,fun方法的定义里面的this确实等同于s,但是在fun方法中调用的是父类的print方法
你可以用下面的代码试试看,当你将Student类中的print方法去掉时,ms中不会出现print
Method[] ms = this.getClass().getDeclaredMethods();
for (Method m : ms) {
m.setAccessible(true);
System.out.println(m.getName());
}
作者: 刘云龙 时间: 2012-7-27 02:27
1Student类的实例对象s,为何能通过fun方法访问到父类中的private方法?
被private修饰,只能被自己调用,这里的调用实际是指直接调用,即直接调用使用。你这里的并不是直接调用,而是通过public的fun间接的调用。这就类似定义private string name这样的私有成员,但我们可以通过public setName()来改变name值一样。
2当s调用fun方法的时候,fun方法的定义里面的this关键字不是应该等同于s么?
你在子类中没有复写父类的fun方法,当你新建一个子类实例时,在内存中是先初始化父类,由于你没有复写fun方法,在调用时,调用的是父类的fun方法,此时的this不是子类的this,而是父类的this,也就相当于子类中的super,故this并不等同于s。
作者: 吴立杰 时间: 2012-7-27 07:32
- class Person
- {
- String str ="Person_field";
- //被final修饰的方法是不可以被子类覆盖的
- final void test2()
- {
- System.out.println("Person_test2");
- }
- //私有的方法,子类继承的时候都它都看不见
- private void print()
- {
- System.out.println("Person");
- }
- public void test()
- {
- System.out.println("person_test");
- }
- public void fun()
- {
- this.test2();//这里调用的是父类的方法,this都是隐士的。
- this.print();//这里调用的是父类的方法,this是隐士的
- this.test();//这里打印出来的结果是子类的。
- }
- }
- class Student extends Person
- {
- String str ="Student_field";
- void print()
- {
- System.out.println("Student");
- }
- public void test()
- {
- System.out.println("Student_test");
- }
- }
- class OverrideDemo04{
- public static void main(String args[])
- {
- Student s = new Student();
- s.fun();
- s.print();
- s.test2();
- Person p =s;
- //p.print();//这个地方连编译都不会通过
- p.test();//这个地方调用的是子类的。
- System.out.println(p.str);//这个地方调用的是父类的字段
- System.out.println(s.str);//这个地方调用的是子类的字段;
- }
- }
复制代码 小结:这个程序的打印结果是
Person_test2
Person
Student_test
Student
Person_test2
Student_test
楼主,其实关于继承中方法的覆盖我有时候也是蒙的,下面只是我推测的想法,希望咱们共同探讨。子类在继承父类的时候,它会跟父类说,把你的东西都给我,父类看到子类手里的extends,也是无可奈何的。但是父类也是有隐私的,保护这个隐私的关键字就是private。有时候有的父类会有自己的传家宝,这个传家宝不想子类继承的时候把这个宝贝改变了,这时候父类就用final修饰了这个宝贝,子类只能用不能覆盖并修改它。对子类继承父类的方法时,在内存中是如何分配的,这个应该时刻往节省内存的角度去考虑,所以方法覆盖的时候就出现不同的地方了。对于父类已有的非私有非final的方法,父类会对子类说,如果你不想修改我给你的方法,那么你继承过去了还得复制一份几乎一模一样的方法,还不如我把这个方法的使用权交给你,你到时候来找我用就行了,这样多省地方,(记住,子类继承父类,父类是不会死的,创建他的个体时,他也是存在的。)而如果子类想修改这些方法(即覆盖,写自己想要的内容),父类就告诉子类,你的权限必须是比我大的,才能覆盖,而且你覆盖了,这个方法就是你的了,跟我没有什么太大的关系,即如果子父类同时被实例化的前提下,这个方法在内方法区里是存在两份的。私有的和final的是很好想的,私有就只有父类的方法区中有,而final是父类的方法区中的这个方法给了子类一个使用的权利,子类可以使用,但是不能修改。这样解释这道题就好解释了:子类继承了父类的方法后,即子类并没有修改父类的方法,那么父类告诉子类,你来我的方法区用吧,子类有这个方法的使用权,但是这个方法得出的结果是什么样的,子类根本就不会知道,因为这个方法不在子类的方法区内。当子类看到自己要调用这个方法的指令后,它拿着父类给的权限去父类的方法区里调用这个方法,而这个方法正好也调用了父类的私有方法,父类说完了,我疏忽了,忘了我这个方法有调用我隐私内容的权利了,但为时已晚,子类看到结果也是大惊,但内心窃喜。而对于子类继承父类后,覆盖了父类的方法,子类调用这些方法当然是子类的方法结果。
多态让我蒙圈了很久,现在说说自己的一些关于多态的想法,子类有和父类一样的print方法,但是这两个是不同的,这个不叫覆盖,因为子类根本就不知道父类有这个方法,此时子类穿上了父类的衣服,调用print方法时,jvm编译时就告诉子类,你无权访问这个方法,说明父类引用变量指向子类对象的时候,调用方法的顺序是先到父类去查找的,而对于子类覆盖后的方法,jvm是知道子类穿着父类的衣服的,所以jvm就告诉子类,你没有能力调用父类的方法,因为你不具备这个功能,所以我就给你个标识,只要你穿着父类的衣服了,我就让你调用自己的衣服了。而对于成员属性,是描述父类和子类特征的,所以jvm再看到子类伪装成父类的时候,它已经有了父类的描述特殊了,即这个子类已经有了调用父类这些成员属性的能力了。写了好久,从五点爬起来,写到现在,楼主,给点分吧?
作者: 罗宵 时间: 2012-7-27 09:40
因为父类中的print方法是private修饰的,是无法继承的所以不能重写的。私有一般都是通过间接访问的,被继承后父类还是存在的。
继承了父类的fun方法,this是代表s,但是父类的fun方法指定访问的是print,所以在子类中也会去访问相应重写的print方法,但是
子类中没有,所以就只能去访问父类了。
作者: 吴立杰 时间: 2012-7-27 10:19
吴立杰 发表于 2012-7-27 07:32
小结:这个程序的打印结果是
Person_test2
Person
楼主,偶不要金钱,偶要技术分哦
作者: 吴立杰 时间: 2012-7-29 16:13
吴立杰 发表于 2012-7-27 07:32
小结:这个程序的打印结果是
Person_test2
Person
嗯,我会努力的,谢啦
作者: 陆强强 时间: 2012-7-29 17:38
本帖最后由 陆强强 于 2012-7-29 17:40 编辑
class Person{
void print(){//去掉private,打印的是Student,所以没有覆盖
System.out.println("Person");
}
public void fun(){
this.print(); //调用父类中的print方法
}
}
class Student extends Person{ //定义子类
void print(){
System.out.println("Student");
}
}
class OverrideDemo04{
public static void main(String args[]){
Student s = new Student();
stu.fun();
}
}
欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) |
黑马程序员IT技术论坛 X3.2 |