2

我正在学习 C++,我对这个练习有疑问。

#include<iostream>
using namespace std;

class B {
    public:
        int x;
        B(int z=1): x(z) {}
};

class D: public B {
    public:
        int y;
        D(int z=5): B(z-2), y(z) {}
};

void fun(B* a, int size) {
    for(int i=0; i<size; ++i) cout << (*(a+i)).x << " ";
}

int main(){
    fun(new D[4],4); cout << "**1\n";
    B* b = new D[4]; fun(b,4); cout << "**2\n";
    b[0] = D(6); b[1] = D(9); fun(b,4); cout << "**3\n";
    b = new B[4]; b[0] = D(6); b[1] = D(9);
    fun(b,4); cout << "**4\n";
}   

它打印:

3 5 3 5 **1
3 5 3 5 **2
4 7 3 5 **3
4 7 1 1 **4

为什么在 **1 和 **2 之前打印 3 5 3 5?我想 3 3 3 3。在 **3 之前我想 5 7 3 3,在 **4 之前我想 4 7 3 3。你能帮我更好地理解为什么会这样打印吗?

提前致谢!

4

2 回答 2

3

不要多态地处理数组。[ 1 ] [ 2 ] 正如您所观察到的,您将获得未定义的行为。幸运的是,您只是看到了错误的结果,但该页面上有一段代码片段显示了崩溃。

于 2013-08-11T11:50:59.177 回答
1

我将从以下行解释输出 3 5 3 5 **1 的第一行:

fun(new D[4],4); cout << "**1\n";

请注意,fun 方法接受 B* 作为参数,它是 D 的超类。我们正在传递一个 D 对象数组。

此外,请注意,当前进到数组中的下一个对象时,我们在 fun 方法中使用了指针算法。B 有一个变量 (x) 和 D 作为两个 (x 和 y),所以 B 是对象 D 的一半大小。我们的对象是 D 而不是 B(尽管它们被切片成 B 但它们作为 D 存储在连续内存中- 数组),所以我们前进了一半。结果,我们实际上是在打印值 x(来自 B),然后是 y 的值(来自 D)等等。

于 2013-08-11T12:09:06.970 回答