模板默认 init,如果类中没有默认构造函数
Template default init, if there no default constructor in a class
我对C++很陌生。我编写了一个函数模板来对向量中的所有元素求和,如下所示:
template<typename T>
inline T sumator(const std::vector<T> &sequence) {
T init = T{};
return accumulate(sequence.begin(), sequence.end(), init);
}
一切都运行良好,直到默认构造函数不声明,而是从例如 2 个参数声明构造函数。 喜欢这个:
class A {
private:
int a;
int b;
public:
// A() = default;
A(int aa, int bb) : a(aa),b(bb){};
A operator+(const A &right) const {
return {a + right.a, b + right.b};
}
bool operator==(const A &right) const {
return (a == right.a && b == right.b);
}
};
vector<A> bc = {{3,5},{5,5}};
A res = Adder::sumator(works);
A result = {8,10};
assert(result,res);
现在我收到一个错误,例如:错误:没有匹配的构造函数来初始化"A" T init = T{};
我怎样才能避免这种情况,不使用 https://en.cppreference.com/w/cpp/types/is_default_constructible
UPD:我有很多类,其中一些带有默认构造函数,有些没有但带有 2 arg 的构造函数,一些用于 3 个,等等。我想好好积累每一种情况
我能想到的可能的解决方案。
- 通过提供初始 vlaue,让用户依赖
std::accumulate
。 - 提供具有初始值的重载
sumator
。如果用户的类型没有默认构造函数,则可以使用初始值调用重载。 - 坚持调用函数的类型具有默认的引体。
template<typename T>
inline T sumator(const std::vector<T> &sequence, T init) {
return accumulate(sequence.begin(), sequence.end(), init);
}
template<typename T>
inline T sumator(const std::vector<T> &sequence) {
return accumulate(sequence.begin(), sequence.end(), {});
}
您可以使用默认参数添加另一个参数。
template<typename T>
inline T sumator(const std::vector<T> &sequence, const T& init = T{}) {
return accumulate(sequence.begin(), sequence.end(), init);
}
对于类型可以是默认构造的,您仍然可以
sumator(some_vector);
并在无法默认构造T
时指定默认值。
sumator(some_vector, A{0, 0});
您可以将其拆分为两个重载,并添加一个static_assert
,以便在有人试图以错误的方式使用它时提供很好的编译错误消息。
仅当类型是默认可构造的类型时,一个重载才有效,并且一个重载对于不那么好的类型以及以下类型有效:
#include <type_traits>
#include <vector>
template<typename T>
T sumator(const std::vector<T>& sequence) {
static_assert(std::is_default_constructible_v<T>);
return std::accumulate(sequence.begin(), sequence.end(), T{});
}
template<typename T>
T sumator(const std::vector<T>& sequence, const T& init) {
return std::accumulate(sequence.begin(), sequence.end(), init);
}
另一种选择是添加一个可用于默认可构造类型的默认参数 - 您还可以使其更通用,使其不仅适用于vector
s,也适用于list
s 等。
template<template<class, class...> class C, class T, class... Ts>
T sumator(const C<T, Ts...>& sequence, const T& init = T{}) {
return std::accumulate(sequence.begin(), sequence.end(), init);
}
相关文章:
- 为什么在没有显式默认构造函数的情况下,将另一个结构封装在联合中作为成员的结构不能编译
- 初始化具有非默认构造函数的std::数组项的更好方法
- 具有默认模板类型的默认构造函数的类型推导
- 如何使用非默认构造函数实例化模板化类
- 有没有一种代码密度较低的方法来使用非默认构造函数初始化数组?
- 声明没有默认构造函数的字段
- 没有默认构造函数作为模板参数的自定义比较器
- C++17 没有默认构造函数的地图放置(私有默认构造函数)
- 使用移动调用对等构造函数unique_ptr默认构造函数
- C++复制构造函数和默认构造函数
- 将向量从 N1 缩小到 N2 项,而不触发默认构造函数并仅使用 move 语义
- 为什么即使我调用参数化构造函数也会调用默认构造函数?
- 具有非默认构造函数的单例类
- 在 C++ 中声明 const 对象需要用户定义的默认构造函数.如果我有一个可变成员变量,为什么不呢?
- 如何处理没有默认构造函数但在另一个构造函数中构造的对象?
- 在C++中使用默认构造函数初始化对象的不同方法
- 在没有默认构造函数的情况下创建的派生对象
- 强制使用默认构造函数对成员进行未初始化的声明
- 使用默认构造函数初始化对象的不同方法
- 创建类类型的动态分配数组,其中类不得具有默认构造函数