关于Java的动态绑定的一个奇怪的问题

桑汤奈伊伏 2015-08-16 03:56:45
import java.util.*;
class Father{
protected void play(int c){
System.out.println("Father's Play with c callled");
}
}

class Son extends Father{
protected void play(char c){
System.out.println("Son's Play with c callled");
}
}



public class Main{
public static void main(String[] args){
Father fd = new Son(); //向上转型
char c = 'c';
//不是说实例方法是调用实例的那个版本吗?这里play明显属于实例方法, 却没有调用到Son的play(char c)方法???
fd.play(c);
//输出为 Father's Play with c callled
}
}
...全文
356 18 打赏 收藏 转发到动态 举报
写回复
用AI写文章
18 条回复
切换为时间正序
请发表友善的回复…
发表回复
rokesw 2015-08-19
  • 打赏
  • 举报
回复
你这个play()叫重载 不叫重写 而且protected声明的属性,方法且只在本类中有效
rokesw 2015-08-19
  • 打赏
  • 举报
回复
你这个play()叫重载 不叫重写
飏飏一蝶 2015-08-17
  • 打赏
  • 举报
回复
引用 7 楼 libertine1993的回复:
[quote=引用 2 楼 nicholasbobo 的回复:] 首先,你的子类和父类虽然有两个相同方法名的方法,但是参数类型不一样,所以子类的方法并不能覆盖父类的方法,其次,你调用play方法时,父类和子类的方法都可以被调用,但是系统会先从父类里去寻找匹配的方法,所以就调用到父类里的方法了
这里我明白没有覆盖, 但是其实我不明白的是, 向上转型的 涉及了覆盖的调用和不涉及覆盖的调用是不是不一样的?具体是怎样的?谢谢![/quote] overload是同一个类里定义的不同参数的同名方法,这才存在转型的问题。override不存在这个,只存在返回值协变,不存在参数转型。
桑汤奈伊伏 2015-08-17
  • 打赏
  • 举报
回复
引用 2 楼 nicholasbobo 的回复:
首先,你的子类和父类虽然有两个相同方法名的方法,但是参数类型不一样,所以子类的方法并不能覆盖父类的方法,其次,你调用play方法时,父类和子类的方法都可以被调用,但是系统会先从父类里去寻找匹配的方法,所以就调用到父类里的方法了
这里我明白没有覆盖, 但是其实我不明白的是, 向上转型的 涉及了覆盖的调用和不涉及覆盖的调用是不是不一样的?具体是怎样的?谢谢!
gukuitian 2015-08-17
  • 打赏
  • 举报
回复
我觉得这玩意不用纠结了, 虽然不算严重,我还是觉得这应该算是个bug了
桑汤奈伊伏 2015-08-17
  • 打赏
  • 举报
回复
引用 14 楼 gukuitian 的回复:
用father去声明对像,就只能调用father存在的方法, 初始的方法表应该是以father为基础的, 再看实例中是否有与覆盖了father 只是这个char很有意思, 定义的是int 参数, 能认识char 同时定义一个int 和char 参数的同名方法, 这样用char调用的时候优先char,, 没有char参数的时候,就走了int 楼主这例子的father里再加个char的同名方法,就该走子类了 [quote=引用 13 楼 libertine1993 的回复:]
动态绑定的过程:
虚拟机提取对象的实际类型的方法表;
虚拟机搜索方法签名;
调用方法。
按照这种绑定的过程, father句柄指向的实例应该是son的实例, 那么实际类型的方法表应该是 son的方法表, 虚拟机接下来搜索方法签名, son的play(char)方法将会完全匹配, 然而son的方法实际上并没有被调用...
[/quote] 茅塞顿开...感谢
gukuitian 2015-08-17
  • 打赏
  • 举报
回复
用father去声明对像,就只能调用father存在的方法, 初始的方法表应该是以father为基础的, 再看实例中是否有与覆盖了father 只是这个char很有意思, 定义的是int 参数, 能认识char 同时定义一个int 和char 参数的同名方法, 这样用char调用的时候优先char,, 没有char参数的时候,就走了int 楼主这例子的father里再加个char的同名方法,就该走子类了
引用 13 楼 libertine1993 的回复:
动态绑定的过程:
虚拟机提取对象的实际类型的方法表;
虚拟机搜索方法签名;
调用方法。
按照这种绑定的过程, father句柄指向的实例应该是son的实例, 那么实际类型的方法表应该是 son的方法表, 虚拟机接下来搜索方法签名, son的play(char)方法将会完全匹配, 然而son的方法实际上并没有被调用...
桑汤奈伊伏 2015-08-17
  • 打赏
  • 举报
回复
引用 10 楼 gukuitian 的回复:
java里char这个类型算是比较特别了. 要是看过class的字节码就会发现,实际上char是直接存储的int数字,本质上就是一个int 直接输出的char时候时候,如果不指定类型,也会当成int输出. [quote=引用 9 楼 libertine1993 的回复:] [quote=引用 8 楼 qq118194716 的回复:] [quote=引用 7 楼 libertine1993的回复:][quote=引用 2 楼 nicholasbobo 的回复:] 首先,你的子类和父类虽然有两个相同方法名的方法,但是参数类型不一样,所以子类的方法并不能覆盖父类的方法,其次,你调用play方法时,父类和子类的方法都可以被调用,但是系统会先从父类里去寻找匹配的方法,所以就调用到父类里的方法了
这里我明白没有覆盖, 但是其实我不明白的是, 向上转型的 涉及了覆盖的调用和不涉及覆盖的调用是不是不一样的?具体是怎样的?谢谢![/quote] overload是同一个类里定义的不同参数的同名方法,这才存在转型的问题。override不存在这个,只存在返回值协变,不存在参数转型。[/quote] 可能是我没表达清楚, 其实我比较不明白的是, 为什么son的实例赋给了father句柄, 调用的时候,调用的是play(int)而不是paly(char)?
import java.util.*;
class Father{
   protected void play(int c){
       System.out.println("Father's Play with int c callled");
   }
}

class Son extends Father{
    protected void play(char c){
       System.out.println("Son's Play with char c callled");
    } 
}


//测试调用的是哪个带参数的play
public class Main{
    public static void main(String[] args){
	    Father fd = new Son(); //将Son的对象句柄赋值给Father句柄
        char c = 'x';		
		fd.play(c);
	}
}
[/quote][/quote] 我觉得这是正解, 但是如果按照你这种说法, 动态绑定依据的就不是对象的实际类型, 而是对象的句柄了.
动态绑定的过程:
虚拟机提取对象的实际类型的方法表;
虚拟机搜索方法签名;
调用方法。
按照这种绑定的过程, father句柄指向的实例应该是son的实例, 那么实际类型的方法表应该是 son的方法表, 虚拟机接下来搜索方法签名, son的play(char)方法将会完全匹配, 然而son的方法实际上并没有被调用...
gukuitian 2015-08-17
  • 打赏
  • 举报
回复
错了,不是数字,有print(char)方法 直接输出的char时候时候,如果不指定类型,也会当成int输出.
sxiaobei 2015-08-17
  • 打赏
  • 举报
回复
通过父类指向子类的实例,只能调用父类有的方法,如果父类的方法没有被overide那么就调用本身属于父类的方法,如果被overide了,那么就调用被overide的方法,你的方法没有被overide,所以调用不到。 其实overide在,java中被称为动态分派,是在程序运行的时候将符号引用转换为直接引用的,一个实例的所有实例方法的实际入口地址是存储在方法区的一张虚方法表中,这张虚方法表也是在类加载阶段进行初始化的,如果一个方法没有被overide那么该方法,的入口地址就是属于父类的方法,如果被覆盖了那么该实际入口地址就替换成为覆盖后的地址了。所以你调用的父类的方法。
gukuitian 2015-08-17
  • 打赏
  • 举报
回复
java里char这个类型算是比较特别了. 要是看过class的字节码就会发现,实际上char是直接存储的int数字,本质上就是一个int 直接输出的char时候时候,如果不指定类型,也会当成int输出.
引用 9 楼 libertine1993 的回复:
[quote=引用 8 楼 qq118194716 的回复:] [quote=引用 7 楼 libertine1993的回复:][quote=引用 2 楼 nicholasbobo 的回复:] 首先,你的子类和父类虽然有两个相同方法名的方法,但是参数类型不一样,所以子类的方法并不能覆盖父类的方法,其次,你调用play方法时,父类和子类的方法都可以被调用,但是系统会先从父类里去寻找匹配的方法,所以就调用到父类里的方法了
这里我明白没有覆盖, 但是其实我不明白的是, 向上转型的 涉及了覆盖的调用和不涉及覆盖的调用是不是不一样的?具体是怎样的?谢谢![/quote] overload是同一个类里定义的不同参数的同名方法,这才存在转型的问题。override不存在这个,只存在返回值协变,不存在参数转型。[/quote] 可能是我没表达清楚, 其实我比较不明白的是, 为什么son的实例赋给了father句柄, 调用的时候,调用的是play(int)而不是paly(char)?
import java.util.*;
class Father{
   protected void play(int c){
       System.out.println("Father's Play with int c callled");
   }
}

class Son extends Father{
    protected void play(char c){
       System.out.println("Son's Play with char c callled");
    } 
}


//测试调用的是哪个带参数的play
public class Main{
    public static void main(String[] args){
	    Father fd = new Son(); //将Son的对象句柄赋值给Father句柄
        char c = 'x';		
		fd.play(c);
	}
}
[/quote]
桑汤奈伊伏 2015-08-17
  • 打赏
  • 举报
回复
引用 8 楼 qq118194716 的回复:
[quote=引用 7 楼 libertine1993的回复:][quote=引用 2 楼 nicholasbobo 的回复:] 首先,你的子类和父类虽然有两个相同方法名的方法,但是参数类型不一样,所以子类的方法并不能覆盖父类的方法,其次,你调用play方法时,父类和子类的方法都可以被调用,但是系统会先从父类里去寻找匹配的方法,所以就调用到父类里的方法了
这里我明白没有覆盖, 但是其实我不明白的是, 向上转型的 涉及了覆盖的调用和不涉及覆盖的调用是不是不一样的?具体是怎样的?谢谢![/quote] overload是同一个类里定义的不同参数的同名方法,这才存在转型的问题。override不存在这个,只存在返回值协变,不存在参数转型。[/quote] 可能是我没表达清楚, 其实我比较不明白的是, 为什么son的实例赋给了father句柄, 调用的时候,调用的是play(int)而不是paly(char)?
import java.util.*;
class Father{
   protected void play(int c){
       System.out.println("Father's Play with int c callled");
   }
}

class Son extends Father{
    protected void play(char c){
       System.out.println("Son's Play with char c callled");
    } 
}


//测试调用的是哪个带参数的play
public class Main{
    public static void main(String[] args){
	    Father fd = new Son(); //将Son的对象句柄赋值给Father句柄
        char c = 'x';		
		fd.play(c);
	}
}
梁金晶 2015-08-16
  • 打赏
  • 举报
回复
五楼解释很专业
飏飏一蝶 2015-08-16
  • 打赏
  • 举报
回复
参数不一致,差评 override都算不上 最多算个overwrite
code小生 2015-08-16
  • 打赏
  • 举报
回复
import java.util.*;
class Father{      
   protected void play(int c){
       System.out.println("Father's Play with c callled");
   }
}

class Son extends Father{	
    protected void play(int c){ //将这里的char改为int就可以了
       System.out.println("Son's Play with c callled");
    } 
}

public class DynamicBindinng{
    public static void main(String[] args){
	    Father fd = new Son(); //向上转型
		char c = 'c';
		//不是说实例方法是调用实例的那个版本吗?这里play明显属于实例方法, 却没有调用到Son的play(char c)方法???
		fd.play(c);
		//输出为 Father's Play with c callled
	}
}
ocean_san 2015-08-16
  • 打赏
  • 举报
回复
父类和子类的方法名字相同且参数类型相同,你是不满足第二个条件。
nicholasbobo 2015-08-16
  • 打赏
  • 举报
回复
首先,你的子类和父类虽然有两个相同方法名的方法,但是参数类型不一样,所以子类的方法并不能覆盖父类的方法,其次,你调用play方法时,父类和子类的方法都可以被调用,但是系统会先从父类里去寻找匹配的方法,所以就调用到父类里的方法了
zzl蓝泐 2015-08-16
  • 打赏
  • 举报
回复
因为子类虽然继承了父类,但是没有重写父类的方法
主要特性Java 语言是简单的:Java 语言的语法与 C 语言和 C++ 语言很接近,使得大多数程序员很容易学习和使用。另一方面,Java 丢弃了 C++ 中很少使用的、很难理解的、令人迷惑的那些特性,如操作符重载、多继承、自动的强制类型转换。特别地,Java 语言不使用指针,而是引用。并提供了自动分配和回收内存空间,使得程序员不必为内存管理而担忧。Java 语言是面向对象的:Java 语言提供类、接口和继承等面向对象的特性,为了简单起见,只支持类之间的单继承,但支持接口之间的多继承,并支持类与接口之间的实现机制(关键字为 implements)。Java 语言全面支持动态绑定,而 C++语言只对虚函数使用动态绑定。总之,Java语言是一个纯的面向对象程序设计语言。Java语言是分布式的:Java 语言支持 Internet 应用的开发,在基本的 Java 应用编程接口中有一个网络应用编程接口(java net),它提供了用于网络应用编程的类库,包括 URL、URLConnection、Socket、ServerSocket 等。Java 的 RMI(远程方法激活)机制也是开发分布式应用的重要手段。Java 语言是健壮的:Java 的强类型机制、异常处理、垃圾的自动收集等是 Java 程序健壮性的重要保证。对指针的丢弃是 Java 的明智选择。Java 的安全检查机制使得 Java 更具健壮性。Java语言是安全的:Java通常被用在网络环境中,为此,Java 提供了一个安全机制以防恶意代码的攻击。除了Java 语言具有的许多安全特性以外,Java 对通过网络下载的类具有一个安全防范机制(类 ClassLoader),如分配不同的名字空间以防替代本地的同名类、字节代码检查,并提供安全管理机制(类 SecurityManager)让 Java 应用设置安全哨兵。Java 语言是体系结构中立的:Java 程序(后缀为 java 的文件)在 Java 平台上被编译为体系结构中立的字节码格式(后缀为 class 的文件),然后可以在实现这个 Java 平台的任何系统中运行。这种途径适合于异构的网络环境和软件的分发。Java 语言是可移植的:这种可移植性来源于体系结构中立性,另外,Java 还严格规定了各个基本数据类型的长度。Java 系统本身也具有很强的可移植性,Java 编译器是用 Java 实现的,Java 的运行环境是用 ANSI C 实现的。Java 语言是解释型的:如前所述,Java 程序在 Java 平台上被编译为字节码格式,然后可以在实现这个 Java 平台的任何系统中运行。在运行时,Java 平台中的 Java 解释器对这些字节码进行解释执行,执行过程中需要的类在联接阶段被载入到运行环境中。Java 是高性能的:与那些解释型的高级脚本语言相比,Java 的确是高性能的。事实上,Java 的运行速度随着 JIT(Just-In-Time)编译器技术的发展越来越接近于 C++。Java 语言是多线程的:在 Java 语言中,线程是一种特殊的对象,它必须由 Thread 类或其子(孙)类来创建。通常有两种方法来创建线程:其一,使用型构为 Thread(Runnable) 的构造子类将一个实现了 Runnable 接口的对象包装成一个线程,其二,从 Thread 类派生出子类并重写 run 方法,使用该子类创建的对象即为线程。值得注意的是 Thread 类已经实现了 Runnable 接口,因此,任何一个线程均有它的 run 方法,而 run 方法中包含了线程所要运行的代码。线程的活动由一组方法来控制。Java 语言支持多个线程的同时执行,并提供多线程之间的同步机制(关键字为 synchronized)。Java 语言是动态的:Java 语言的设计目标之一是适应于动态变化的环境。Java 程序需要的类能够动态地被载入到运行环境,也可以通过网络来载入所需要的类。这也有利于软件的升级。另外,Java 中的类有一个运行时刻的表示,能进行运行时刻的类型检查。

62,614

社区成员

发帖
与我相关
我的任务
社区描述
Java 2 Standard Edition
社区管理员
  • Java SE
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧