如何根据参数在复制构造函数和默认构造函数之间切换
How to switch between copy-constructor and default constructor depending on argument?
我有以下代码:
struct S
{
S(): a(42) {}
int a;
};
class P
{
public:
P(S const *s): m_s(s ? *s : /*init m_s by default ctor - how to achieve it?*/)
private:
S m_s;
};
我希望m_s
由复制构造函数或默认构造函数初始化,具体取决于指针s
:
P p1(nullptr); // expect default ctor: p1.m_s.a = 42
S s;
s.a = 84;
P p2(&s); // expect copy ctor: p2.m_s.a = 84
我怎样才能用最优雅的方式来做呢?
您可以只写:
class P {
public:
P(S const *s): m_s(s ? *s : S()){}
private:
S m_s;
};
您可能会发现将条件提取到一个单独的辅助函数更优雅:
S createS(S const *s) {
return s ? *s : S();
}
在nullptr
参数的情况下,这看起来可能会执行不必要的复制,但实际上编译器会执行RVO并优化复制。
现场演示。
一个仅适用于空指针常量的部分解决方案是添加一个重载:
P(std::nullptr_t) : m_s() {}
P(const S * s) : m_s(AssertNotNull(s)) {}
这里我使用:
template <typename T> T * AssertNotNull(T * p) {
if (!p) std::abort();
return p;
}
如果你需要一个动态开关,你可以使用一个辅助功能:
struct P {
static const S & maybe(const S * s) {
static S x;
return s ? *s : x;
}
P(const S * s) : m_s(maybe(s)) {}
S m_s;
}
这实际上并没有在复制和默认构造函数之间切换(因为你不能动态地这样做),但如果指针为null,它会通过从默认构造的实例复制来伪造效果。
相关文章:
- 为什么在没有显式默认构造函数的情况下,将另一个结构封装在联合中作为成员的结构不能编译
- 初始化具有非默认构造函数的std::数组项的更好方法
- 具有默认模板类型的默认构造函数的类型推导
- 如何使用非默认构造函数实例化模板化类
- 为什么不调用移动构造函数?(默认情况下只有构造器,没有别的)
- 有没有一种代码密度较低的方法来使用非默认构造函数初始化数组?
- 声明没有默认构造函数的字段
- 基类中的默认析构函数禁用子类中的移动构造函数(如果有成员)
- 没有默认构造函数作为模板参数的自定义比较器
- C++17 没有默认构造函数的地图放置(私有默认构造函数)
- 使用移动调用对等构造函数unique_ptr默认构造函数
- C++复制构造函数和默认构造函数
- 将向量从 N1 缩小到 N2 项,而不触发默认构造函数并仅使用 move 语义
- 构造函数默认公共和私有变量
- 类模板构造函数默认参数
- 构造函数默认参数
- C++模板构造函数默认参数
- 在c++中设置构造函数默认值
- c++构造函数默认参数
- C++构造函数默认值头文件