当前位置 博文首页 > m0_48462648的博客:面向对象的超级面试题,涉及封装多态继承等
虽然总是我先头也不回说走就走,但有时候也会觉得,自己才是每段已经结束的关系里,被放弃的那个人。
废话少说上代码:
public class Test1 {
static class A{
public String run(D obj) {
return ("A & D");
}
public String run(A obj){
return ("A & A");
}
}
static class B extends A{
public String run(B obj) {
return ("B & B");
}
public String run(A obj){
return ("B & A");
}
}
static class C extends B {}
static class D extends B {}
static class E{
public static void main(String[] args) {
A aa = new A();
A ab = new B();
B b = new B();
C c = new C();
D d = new D();
System.out.println(aa.run(b));
System.out.println(aa.run(c));
System.out.println(aa.run(d));
System.out.println(ab.run(b));
System.out.println(ab.run(c));
System.out.println(ab.run(d));
System.out.println(b.run(b));
System.out.println(b.run(c));
System.out.println(b.run(d));
}
}
}
以上代码来自知乎:如有侵犯,请告知,原文链接:点击进入原文章
不要用电脑,纯脑子想,这个输出的内容很简单,说出打印的结果是什么?3分钟时间:
321!ok,这个面试题对于一些基础不是特别好以及对刚接触面向对象的人,他肯定是会疯掉,如果此时此刻你已经算出结果了,那对不起:
如果没有算出来的童鞋,请继续往下看,这就是结果:
A & A
A & A
A & D
B & A
B & A
A & D
B & B
B & B
A & D
为了让大家方便观看代码,我现在把结果和源代码放在一起:
public class Test1 {
static class A{
public String run(D obj) {
return ("A & D");
}
public String run(A obj){
return ("A & A");
}
}
static class B extends A{
public String run(B obj) {
return ("B & B");
}
public String run(A obj){
return ("B & A");
}
}
static class C extends B {}
static class D extends B {}
static class E{
public static void main(String[] args) {
A aa = new A();
A ab = new B();
B b = new B();
C c = new C();
D d = new D();
System.out.println(aa.run(b)); //1, A & A
System.out.println(aa.run(c)); //2, A & A
System.out.println(aa.run(d)); //3, A & D
System.out.println(ab.run(b)); //4, B & A
System.out.println(ab.run(c)); //5, B & A
System.out.println(ab.run(d)); //6, A & D
System.out.println(b.run(b)); //7, B & B
System.out.println(b.run(c)); //8, B & B
System.out.println(b.run(d)); //9, A & D
}
}
}
OK,那我们一个一个分析,首先分析一下整个程序,在A类里面有两个方法,分别是:
这两个方法虽然名字一样,但是参数类型不一样,一个是D类的参数,一个是A类的参数;
接着B类继承了A类:此时B类不仅可以调用自己的方法,还可以调用A类的方法:
随后,看下面代码:
此时他们的关系应该是:
C、D类继承了B类,B类继承了A类;
C、D类可以调用B、A类的方法,B类可以调用A类的方法:
测试类中:
第一个属于创建了A类的对象,对象名为aa
第二个,使用了多态,在A类内编译调用,最后输出B类的内容
最后三个,都是创建了自身类的对象
我们一个一个输出语句来分析:
System.out.println(aa.run(b)); //1, A & A
首先,这个是调用A类的aa方法,但是在aa方法中,传入了b,b是B类的对象,首先,我们在A类里面找,有没有方法参数是B类型的:
我们发现,A类方法中,只有D类型的参数和A类型的参数,那我们知道A是B的父类,如果一个方法中子类找不到,那么他就会找父类中找
比如以下例子:
//父类
public class FU {
public void s(String a){
System.out.println("s");
}
}
//子类
public class ZI extends FU{
public void s(int a){
System.out.println("z");
}
}
//测试
public class Test {
public static void main(String[] args) {
ZI zi =new ZI();
zi.s("张三");
}
}
在上面的离职中,子类继承父类,通过测试类new了子类的对象,并用子类调用了s方法,随后传了一个参数”张三“,很明显,我们是使用子类调用的,但是张三它属于字符串变量,系统会首先去找参数是字符串张三的s方法,子类没有,才会去找父类,最后输出s,这样写也许你们看不懂,但换个方式你们就知道了
//父类
public class FU {
public String s(FU fu){
String i="张三";
return i;
}
}
//子类
public class ZI extends FU{
}
//测试
public class Test {
public static void main(String[] args) {
FU fu=new FU();
ZI zi = new ZI();
System.out.println(fu.s(zi));
}
}
那现在我们再看,我们在测试类中,使用父类的对象调用s方法,传入了子类的对象,那么我们方法在运行的时候,就会先找子类,但是子类没有,担忧因为子类继承了父类,如果子类没有,就会找父类
第一步,先去传入zi的本身的对象里面找,没有,那就去父类里面找,那此时对象就换了,从之前的zi对象换成了fu对象
那我们接着看源代码:
System.out.println(aa.run(b));
去父类找之后,更改成父类的对象,这个方法就变成了
System.out.println(aa.run(a));