2

我正在使用练习考试学习 C++ 考试。我在确定何时调用某些构造函数时遇到问题。代码是 -

using namespace std;
#include <iostream>

class Fraction
{
  private:
   int numerator, denominator;
  public:
   Fraction( int = 0, int = 1 );
   friend ostream& operator<<( ostream&, const Fraction& );
  };

 void debug( Fraction, Fraction );
 Fraction task( Fraction&, const Fraction& );

int main()
  {
    Fraction * A[6], * B; // Line 1
    Fraction C( 3 ), D( C ); // Line 2
    Fraction E[4], F = D; // Line 3
    Fraction G, H( 3, 4 ); // Line 4
    debug( C, D ); // Line 5
    B = new Fraction; // Line 6
    F = task( H, C );
    delete B;
    return 0;
  }

 void debug( Fraction X, Fraction Y )
 {
   cout << X << endl << Y << endl;
 }

 Fraction task( Fraction& X, const Fraction& Y )
 {
   Fraction Z;
   Z = Fraction( 5, 2 ); // Line 7
   return 1; // Line 8
 }

现在,我认为在 3 种类型的实例中调用了默认的复制构造函数。

示例 1 -

        Fraction A(1,2);

        Fraction B(A);

示例 2 -

        Fraction A(1,2);

        Fraction A = B;

3 - 从函数返回对象时

4 - 当对象作为参数传递给函数时

有两个问题我似乎无法回答——

问题 1 - 考虑上面的源代码。当标记为“第 3 行”和“第 4 行”的语句被编译和执行时,对于这些语句中的每一个,将对“分数”类的构造函数进行多少次调用?

答案是 - D) 线路 3:5 次呼叫线路 4:2 次呼叫

这个问题的答案是 D。但是,当我回答它时,我得到了不同的结果 -

    Fraction E[4], F = D; // Line 3

我看到数组 E[4] 的默认构造函数被调用了 5 次,但我认为这个语句也相当于 -

    Fraction E[4], Fraction F(D);

因此,默认构造函数被调用 5 次,复制构造函数被调用一次。显然,这是不正确的,因为构造函数只被调用了 5 次。我没有看到我的错误在哪里。

我的另一个问题是以下问题-

  1. 考虑图 2 中的源代码(上一页)。当标记为“第 7 行”和“第 8 行”的语句被编译并执行时,对于这些语句中的每一个,将对“分数”类的构造函数进行多少次调用?

    答案是 - D) 7 号线:1 次呼叫 8 号线:1 次呼叫

        Fraction task( Fraction& X, const Fraction& Y )
        {
          Fraction Z;
          Z = Fraction( 5, 2 ); // Line 7
          return 1; // Line 8
        }
    

    我原本以为会调用默认构造函数来为 Fraction(5,2) 创建一个临时对象,然后复制构造函数会将 Fraction(5, 2) 复制到 Z。这实际上是默认赋值运算符的操作吗?至于第 8 行,这是否调用了复制构造函数,因为它返回了 Fraction 类型的对象?

我在这里的时候还有几个问题-

将 -

debug(B, C)

调用复制构造函数两次以传递分数对象 B 和 C 的副本?

并且是 -

F = task( H, C );

调用默认赋值运算符以及 F 已经作为初始化的 Fraction 对象存在?

4

1 回答 1

1

您基本上想知道何时调用默认构造函数和复制构造函数。

对于你的第一个问题:

Fraction E[4], F = D; // Line 3

E[4]是一个Fraction对象数组,它调用默认构造函数 4 次,因为您没有显式调用任何其他构造函数来初始化这些对象。由于在第 2 行D已经构造了对象,因此F是使用复制构造函数构造的,在这种情况下,编译器会为您生成一个,因为您没有为自己定义一个。所以在这种情况下,有 4 个默认构造函数调用和 1 个复制构造函数调用。在第 4 行,两个对象都调用了默认构造函数,因此在默认 ctor 上调用了两次。

debug(B, C)
call the copy constructor twice to pass copies of Fraction objects B and C?

是的,因为参数是在函数中按值传递的,所以对两者都debug调用了复制构造函数BC

一般来说,复制构造函数在以下情况下被调用:

  1. 使用现有对象(如F= DF(D))实例化一个对象时
  2. 按值传递对象时。
  3. 当对象按值从函数返回时。

希望这可以为您提供足够的信息来理解代码。

于 2013-03-24T02:10:44.023 回答