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

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

  •  
  •   diangdiang · 2017-10-05 21:22:21 +08:00 · 3099 次点击
    这是一个创建于 2602 天前的主题,其中的信息可能已经有所发展或是发生改变。
    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
    diangdiang
        1
    diangdiang  
    OP
       2017-10-05 21:24:35 +08:00
    校招笔试题,搞不懂啊,谢谢 ~
    rogerchen
        2
    rogerchen  
       2017-10-05 21:31:33 +08:00
    a2 中 B and B 对应的 overloading 不参与 resolution
    diangdiang
        3
    diangdiang  
    OP
       2017-10-05 21:34:13 +08:00
    为啥不参与呢,能详细点吗?小白求助
    rogerchen
        4
    rogerchen  
       2017-10-05 21:36:41 +08:00
    @diangdiang a2 的类型是 A,A 没有 public String show(B obj) 这个方法。
    diangdiang
        5
    diangdiang  
    OP
       2017-10-05 21:37:31 +08:00
    哎呀呀,妙啊!多谢大佬
    ioc
        6
    ioc  
       2017-10-05 23:44:30 +08:00 via Android
    考重载和多态的混淆?
    sosloop
        7
    sosloop  
       2017-10-05 23:54:06 +08:00 via Android
    就是考多态,函数重载和重写
    jimisun
        8
    jimisun  
       2017-10-06 00:08:16 +08:00 via Android
    @rogerchen 既然 A 类型没有 public String show(B obj 这个方法 为什么 a2.show(b);还能传入 b 类型的参数呢?
    lihongjie0209
        9
    lihongjie0209  
       2017-10-06 00:26:28 +08:00
    这是函数重写, 多态(动态绑定)的基础是有一个类型(interface), 然后有不同的类(class)来实现, 最后在运行时(runtime)绑定.
    Spectre
        10
    Spectre  
       2017-10-06 00:35:20 +08:00
    @jimisun 同疑惑
    Sikoay
        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",不知道这样解释对不对
    Sikoay
        12
    Sikoay  
       2017-10-06 00:39:00 +08:00 via Android
    @jimisun
    @Spectre

    因为 B 为 A 的子类啊,所以即使 show(B obj)丢失之后也能够传入 B,但是此时传入的 B 已经被向上转型为 A 了
    Spectre
        13
    Spectre  
       2017-10-06 00:53:30 +08:00
    @Sikoay B 继承于 A 能传 A 的都能传 B 好像是哎
    Sikoay
        14
    Sikoay  
       2017-10-06 00:56:19 +08:00 via Android
    @Spectre 是的,但是在传递 A 的方法中就不能调用 B 的方法,但是还可以调用 B 重载的 A 的方法,并且能够正常运行,这就是运行时绑定
    wenzhoou
        15
    wenzhoou  
       2017-10-06 07:55:12 +08:00 via Android
    编译后 调用的是 invokevirtual instance:a2 method:show ( A )
    编译时候就决定了一部分。运行时候决定了另一部分。
    Doodlister
        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 类型作为参数。
    guodong110
        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
    HackerOO7
        18
    HackerOO7  
       2017-10-06 10:03:16 +08:00
    声明 a2 的 A 类是静态类型,B 是实际类型,经过动态绑定后,调用的是实际实际类型中的方法
    loopback
        19
    loopback  
       2017-10-06 21:46:19 +08:00
    静态多分派(两个宗量)
    动态单分派(一个宗量)
    vwok
        20
    vwok  
       2017-10-07 00:39:13 +08:00 via Android
    diangdiang
        21
    diangdiang  
    OP
       2017-10-07 15:01:01 +08:00
    多谢各位热心 V 友!
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   实用小工具   ·   1194 人在线   最高记录 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 23ms · UTC 18:38 · PVG 02:38 · LAX 10:38 · JFK 13:38
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.