- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我有一些 C++ 背景知识,也了解一些 Java(显然远远不够)。
当我看到 Java 或 C++ 中的覆盖行为时,它们似乎没有太大区别。在 JAVA 中给出以下示例:
class Animal{
public void move(){
System.out.println("Animals can move");
}
}
class Dog extends Animal{
public void move(){
System.out.println("Dogs can walk and run");
}
}
public class TestDog{
public static void main(String args[]){
Animal a = new Animal(); // Animal reference and object
Animal b = new Dog(); // Animal reference but Dog object
a.move();// runs the method in Animal class
b.move();//Runs the method in Dog class
}
}
在Java中,你使用基类引用,在C++中你使用基类指针,根据它指向的实例类型(基类对象实例或子类实例),你可以实现多态性。
以上是基于你使用基类引用或指针调用实例方法,对吧?
现在我在 Java 中看到了这个例子。
What is the order of the Constructors in this Java Code?
基本上就是说如果一个基类的函数被重写了,那么在创建子类对象的过程中,连基类的初始化部分都会受到影响。请参阅我从上面的链接复制的以下解释:
new Son()
=>
Son._init
=> first every constructor calls super()
Father._init
Object._init
who() => is overridden, so prints "son" !!!!!
tell(name) => name is private, so cannot be overridden => "father"
who() => "son"
tell(name) => "son"
为什么会这样?我的意思是这是否符合应该如何使用多态性?在初始化基类部分时,为什么要使用子类的重写函数?
在 Java 文档中 http://docs.oracle.com/javase/specs/jls/se7/html/jls-12.html#jls-12.5 ,我只找到这个:
“与 C++ 不同,Java 编程语言在创建新类实例期间不会指定更改的方法分派(dispatch)规则。如果调用的方法在被初始化对象的子类中被覆盖,则使用这些覆盖方法,甚至在新对象完全初始化之前。”
但我不知道背后的原因,感觉很奇怪。
有什么想法吗?
最佳答案
这是 C++ 比 Java 更努力保护您免于搬起石头砸自己脚的极少数情况之一。 (或者至少它有这样做的崇高意图。)
如果您尝试从 B 的构造函数中调用基类 B 的可重写(虚)方法 M,您很可能用任何语言搬起石头砸自己的脚。这是因为 M 很可能在派生类中被重写D,但是在构建B的时候,D还没有构建。因此,在调用 D 的构造函数之前调用 D.M。这可能意味着灾难。
因此,Java 允许这种情况发生,使用风险自负。 (如果您启用了足够多的警告,您的编译器会告诉您您的生活很危险。)
C++ 也不禁止这样做,但它稍微改变了它的行为以包含损害,可以这么说:当你从构造函数中调用虚方法时,它并没有真正将它作为虚方法调用,(使用 VMT 查找,)但它直接调用它,作为一种非虚拟方法。
(要么,要么在 B 的构造函数中,它只是使用类 B 的 VMT 而不是 D 的 VMT。现在想想,这是有道理的。但我不确定,这已经很长时间了自从我上次解决 C++ 的这种行为以来已经过去了。)
关于java - 为什么重写在 JAVA 中的工作方式与 C++ 有所不同?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34019439/
我现在遇到了一个奇怪的问题。 我正试图解决 Chrome 中的一个 CSS 错误,但每当我进入开发者工具时,该错误就神秘地修复了。 遇到这种情况你会怎么处理? 为什么渲染引擎要捉弄我? ;) 最佳答案
我正在学习 APIM 政策。在缓存策略中,有许多 最佳答案 我猜你指的是https://learn.microsoft.com/en-us/azure/api-management/api-manag
我正在学习 APIM 政策。在缓存策略中,有许多 最佳答案 我猜你指的是https://learn.microsoft.com/en-us/azure/api-management/api-manag
如果'Test'是一个普通类,有什么区别: Test* test = new Test; 和 Test* test = new Test(); 最佳答案 让我们学究气点,因为有些差异实际上会影响代码的
如果'Test'是一个普通类,有什么区别: Test* test = new Test; 和 Test* test = new Test(); 最佳答案 让我们学究气点,因为有些差异实际上会影响代码的
如果'Test'是一个普通类,有什么区别: Test* test = new Test; 和 Test* test = new Test(); 最佳答案 让我们学究气点,因为有些差异实际上会影响代码的
如果'Test'是一个普通类,有什么区别: Test* test = new Test; 和 Test* test = new Test(); 最佳答案 让我们学究气点,因为有些差异实际上会影响代码的
如果'Test'是一个普通类,有什么区别: Test* test = new Test; 和 Test* test = new Test(); 最佳答案 让我们学究气点,因为有些差异实际上会影响代码的
我有一些 C++ 背景知识,也了解一些 Java(显然远远不够)。 当我看到 Java 或 C++ 中的覆盖行为时,它们似乎没有太大区别。在 JAVA 中给出以下示例: class Animal{
我有一个index.html 文件,引用了一个javascript 文件 asd 在我的index.js function init() {
我正在尝试构建一个 AWS Signature v4 Auth header 来调用 STS GetCallerIdentity API 根据文档 https://docs.aws.amazon.co
本质上,我想根据图库中每个图像的尺寸更改图像裁剪的变量。如果图像的高度大于宽度,我想切换到高度裁剪,否则保留宽度裁剪。到目前为止,我正在使用 Galleria 插件,我有以下代码: if ($(ima
我已经缩小了我的基本问题范围:从 IE9 到 FireFox(和 Chrome),数组的范围有所不同。 在以下函数(仅摘录)中,我声明一个数组,然后通过调用 $.getJSON() 来填充它。因为引用
我创建了一类消息,包括内容和发件人。我成功地将所需的数据存储在 Parse 中,并且正在查询它们。到目前为止,没有问题。然后,我尝试根据发送者或接收者过滤消息,以便在我的 tableView 上以不同
大家好,我有一个奇怪的问题 - 我正在视频播放器中播放视频。用于相同的代码是: let url = URL(string: "xxxxxxxxxx.mp4") player = AVPlaye
这是我的 json 输入 { "students_key": { "student_key_one": { "profile_root": "/profile/student_
我是一名优秀的程序员,十分优秀!