c++ - 虚函数C++中的默认参数

标签 c++ inheritance virtual-functions

我阅读了 C++ 中的继承机制和虚函数。

根据我的知识(在我遇到的所有示例中),继承的方法具有与父类相同的签名。

我的问题如下: 我知道函数默认参数值不是函数签名的一部分。

我能否在父类虚函数中将此值定义为某个常量,并在派生类中声明并实现没有此默认值的重写方法。

在这种情况下,当我使用指向父类的指针调用派生对象的方法时,是否会在有/没有此默认初始化的情况下调用该函数?

谢谢

最佳答案

默认参数主要是语法糖,在编译时确定。另一方面,虚拟分派(dispatch)是一种运行时特性。选择与实际调用的函数一起定义的默认参数可能并不令人惊讶,但由于上述原因,这是不可能的(至少在没有额外的运行时开销的情况下)。

因此,默认参数由编译器使用调用成员函数的对象的 static 类型选择。让我们看一个例子。

#include <iostream>
#include <memory>

class Base
{

public:

  virtual void
  f(int a, int b = 1)
  {
    std::cout << "Base: a = " << a << ", b = " << b << "\n";
  }
};

class Derived : public Base
{

public:

  virtual void
  f(int a = 1, int b = 2) override
  {
    std::cout << "Derived: a = " << a << ", b = " << b << "\n";
  }
};

int
main()
{
  std::unique_ptr<Base> base_as_base {new Base {}};
  std::unique_ptr<Base> derived_as_base {new Derived {}};
  std::unique_ptr<Derived> derived_as_derived {new Derived {}};
  base_as_base->f(0);        // Base:    a = 0, b = 1
  derived_as_base->f(0);     // Derived: a = 0, b = 1
  // derived_as_base->f();   // compiler error
  derived_as_derived->f(0);  // Derived: a = 0, b = 2
  derived_as_derived->f();   // Derived: a = 1, b = 2
}

我同意这令人困惑。请不要写这样的代码。幸运的是,有一个简单的解决方法。除了完全不使用默认参数外,我们还可以使用一个叫做非虚拟接口(interface) 的习惯用法。虚函数是 protected 并且没有给定任何默认参数。然后它只能由基类中的非 virtual 函数间接调用。该函数可以在一个地方定义所有默认参数。

#include <iostream>
#include <memory>

class Base
{

public:

  void
  f(int a, int b = 1)
  {
    this->impl(a, b);
  }

protected:

  virtual void
  impl(int a, int b)
  {
    std::cout << "Base: a = " << a << ", b = " << b << "\n";
  }
};

class Derived : public Base
{

protected:

  virtual void
  impl(int a, int b) override
  {
    std::cout << "Derived: a = " << a << ", b = " << b << "\n";
  }
};

int
main()
{
  std::unique_ptr<Base> base_as_base {new Base {}};
  std::unique_ptr<Base> derived_as_base {new Derived {}};
  std::unique_ptr<Derived> derived_as_derived {new Derived {}};
  base_as_base->f(0);        // Base:    a = 0, b = 1
  derived_as_base->f(0);     // Derived: a = 0, b = 1
  derived_as_derived->f(0);  // Derived: a = 0, b = 1
}

关于c++ - 虚函数C++中的默认参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27974653/

相关文章:

c++ - 使用虚函数的力量

c++ - cpp继承问题

c++ - 什么时候使用 exit() 与 C++ 中的异常比较好?

maven - Maven的聚合模型如何遵守其依赖机制?

python - 初始化子类中在父类中使用的属性

java - 在层次结构的中间类中使用 Hibernate 版本控制

c++ - 缺少 MSVCP100D.dll (OpenCV) Visual Studio 2013

c++ - 如何列出 C++ 数组中所有可能的组合总和?

带有 istream& 参数的 C++ 函数

delphi - 编译器允许调用同级类中的 protected 方法,但调用基类