28

可变参数构造函数是否应该隐藏隐式生成的构造函数,即默认构造函数和复制构造函数?

struct Foo
{
    template<typename... Args> Foo(Args&&... x)
    {
        std::cout << "inside the variadic constructor\n";
    }
};

int main()
{
    Foo a;
    Foo b(a);
}

不知何故,我希望在阅读此答案后不会打印任何内容,但它会inside the variadic constructor在 g++ 4.5.0 上打印两次 :( 这种行为正确吗?


在没有可变参数模板的情况下也会发生这种情况:

struct Foo
{
    Foo()
    {
        std::cout << "inside the nullary constructor\n";
    }

    template<typename A> Foo(A&& x)
    {
        std::cout << "inside the unary constructor\n";
    }
};

int main()
{
    Foo a;
    Foo b(a);
}

同样,两行都被打印出来。

4

1 回答 1

21

事实上,隐式声明的复制构造函数的声明并没有被禁止。由于重载决议的规则,它只是没有被调用。

隐式声明的复制构造函数具有Foo(const Foo&). 重要的部分是它需要一个 const 引用。您的构造函数模板采用非常量引用。

a不是 const,因此非 const 用户声明的构造函数模板优于隐式声明的复制构造函数。要调用隐式声明的复制构造函数,可以使用aconst:

const Foo a;
Foo b(a);

或者您可以使用static_cast来获取对以下内容的 const 引用a

Foo a;
Foo b(static_cast<const Foo&>(a));

描述这一点的重载解决规则主要在 C++0x FCD 的 §13.3.3.2/3 中找到。这个特殊的场景,结合了左值和右值引用,在第 303 页上的各种示例中有所描述。


可变参数构造函数模板将抑制隐式声明的默认构造函数,因为可变参数构造函数模板是用户声明的,并且仅在没有用户声明的构造函数时才提供隐式声明的默认构造函数(C++0x FCD §12.1/5):

如果 class 没有用户声明X的构造函数,则没有参数的构造函数被隐式声明为默认值。

可变参数构造函数模板不会抑制隐式声明的复制构造函数,因为只有非模板构造函数可以是复制构造函数(C++0x FCD §12.8/2、3 和 8):

类的非模板构造函数X是复制构造函数,如果它的第一个参数是 , 或 类型,或者X&没有其他参数,或者所有其他参数都有默认参数。const X&volatile X&const volatile X&

一个类的非模板构造函数X是一个移动构造函数,如果它的第一个参数是 , , , or 类型X&&const X&&并且volatile X&&要么const volatile X&&没有其他参数,要么所有其他参数都有默认参数。

If the class definition does not explicitly declare a copy constructor and there is no user-declared move constructor, a copy constructor is implicitly declared as defaulted.

于 2010-06-01T22:50:37.270 回答