当前位置 博文首页 > m0_48462648的博客:面向对象的超级面试题,涉及封装多态继承等

    m0_48462648的博客:面向对象的超级面试题,涉及封装多态继承等

    作者:[db:作者] 时间:2021-08-02 09:50

    虽然总是我先头也不回说走就走,但有时候也会觉得,自己才是每段已经结束的关系里,被放弃的那个人。

    废话少说上代码:
    在这里插入图片描述

    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));