首页   注册   登录
V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
V2EX  ›  程序员

有 Java 大佬在吗?遇到一个 动态绑定的问题,求指教

  •  
  •   diangdiang · 2017-10-05 21:22:21 +08:00 · 2062 次点击
    这是一个创建于 794 天前的主题,其中的信息可能已经有所发展或是发生改变。
    class A {
    public String show(A obj) {
    	return ("A and A");
    }
    

    }

    class B extends A {

    public String show(B obj) {
    	return ("B and B");
    }
    
    public String show(A obj) {
    	return ("B and A");
    }
    

    }

    public class OverrideTest {

    public static void main(String[] args) {
    
    	A a2 = new B();
    	B b = new B();
    	
    	String res = a2.show(b);
    	System.out.println(res); // 感觉应该输出 B and B,实际输出 B and A
    }
    

    }

    21 回复  |  直到 2017-10-07 15:01:01 +08:00
        1
    diangdiang   2017-10-05 21:24:35 +08:00
    校招笔试题,搞不懂啊,谢谢 ~
        2
    rogerchen   2017-10-05 21:31:33 +08:00
    a2 中 B and B 对应的 overloading 不参与 resolution
        3
    diangdiang   2017-10-05 21:34:13 +08:00
    为啥不参与呢,能详细点吗?小白求助
        4
    rogerchen   2017-10-05 21:36:41 +08:00
    @diangdiang a2 的类型是 A,A 没有 public String show(B obj) 这个方法。
        5
    diangdiang   2017-10-05 21:37:31 +08:00
    哎呀呀,妙啊!多谢大佬
        6
    ioc   2017-10-05 23:44:30 +08:00 via Android
    考重载和多态的混淆?
        7
    chenpipguge   2017-10-05 23:54:06 +08:00 via Android
    就是考多态,函数重载和重写
        8
    jimisun   2017-10-06 00:08:16 +08:00 via Android
    @rogerchen 既然 A 类型没有 public String show(B obj 这个方法 为什么 a2.show(b);还能传入 b 类型的参数呢?
        9
    lihongjie0209   2017-10-06 00:26:28 +08:00
    这是函数重写, 多态(动态绑定)的基础是有一个类型(interface), 然后有不同的类(class)来实现, 最后在运行时(runtime)绑定.
        10
    Spectre   2017-10-06 00:35:20 +08:00
    @jimisun 同疑惑
        11
    Sikoay   2017-10-06 00:35:41 +08:00 via Android
    a2 向上转型为 A,这时 show( B obj)方法丢失,然后调用 a2.show(b)时,由于 a2 中只有重载 show(A obj)过的方法,即调用此方法,输出"B and A",不知道这样解释对不对
        12
    Sikoay   2017-10-06 00:39:00 +08:00 via Android
    @jimisun
    @Spectre

    因为 B 为 A 的子类啊,所以即使 show(B obj)丢失之后也能够传入 B,但是此时传入的 B 已经被向上转型为 A 了
        13
    Spectre   2017-10-06 00:53:30 +08:00
    @Sikoay B 继承于 A 能传 A 的都能传 B 好像是哎
        14
    Sikoay   2017-10-06 00:56:19 +08:00 via Android
    @Spectre 是的,但是在传递 A 的方法中就不能调用 B 的方法,但是还可以调用 B 重载的 A 的方法,并且能够正常运行,这就是运行时绑定
        15
    wenzhoou   2017-10-06 07:55:12 +08:00 via Android
    编译后 调用的是 invokevirtual instance:a2 method:show ( A )
    编译时候就决定了一部分。运行时候决定了另一部分。
        16
    Doodlister   2017-10-06 09:17:10 +08:00
    B 是 A 的子类 A a2 = new B(); 是 B 向父类向上转型, 所以只能调用父类中声明的方法。即 show(A obj)。
    a2.show(b); 这样调用也是没毛病的 因为 b 是 A 的子类对象 所以 可以向上转型为 A 类型作为参数。
        17
    guodong110   2017-10-06 09:47:43 +08:00 via Android
    承链中对象方法的调用存在一个优先级:this.show(O)、super.show(O)、this.show((super)O)、super.show((super)O)。这里 this.show((super)O)满足,又由于在子类向上转型的时候,如果调用的方法是非静态方法且被子类覆写了,那么程序将会调用的是子类的方法,故输出 B and A
        18
    HackerOO7   2017-10-06 10:03:16 +08:00
    声明 a2 的 A 类是静态类型,B 是实际类型,经过动态绑定后,调用的是实际实际类型中的方法
        19
    loopback   2017-10-06 21:46:19 +08:00
    静态多分派(两个宗量)
    动态单分派(一个宗量)
        20
    vwok   2017-10-07 00:39:13 +08:00 via Android
        21
    diangdiang   2017-10-07 15:01:01 +08:00
    多谢各位热心 V 友!
    关于   ·   FAQ   ·   API   ·   我们的愿景   ·   广告投放   ·   感谢   ·   实用小工具   ·   2249 人在线   最高记录 5043   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.3 · 32ms · UTC 15:11 · PVG 23:11 · LAX 07:11 · JFK 10:11
    ♥ Do have faith in what you're doing.