由于 unordered_map 与结构作为键一起使用,显式默认的默认构造函数被隐式删除
Explicitly defaulted default constructor is implicitly deleted because of unordered_map used with a struct as key
我有以下图表 class:
class Graph {
private:
struct Edge {
string vertex1{}, vertex2{};
int val{};
Edge() = default;
~Edge() = default;
explicit Edge(string v1, string v2, int value) : vertex1(std::move(v1)), vertex2(std::move(v2)), val(value) {};
bool operator==(const Edge&) const;
};
unordered_map<Edge, Edge*> edges;
public:
Graph() = default;
~Graph();
}
当我想用默认构造函数构造一个图时,它说Explicitly defaulted default constructor of 'Graph' is implicitly deleted because field 'edges' has a deleted default constructor
。我应该如何更改我的代码,以便能够使用 Graph 的默认构造函数?
unordered_map
中的 Key 需要可哈希。通过添加 std::hash
class 模板的特化,或者在创建 unordered_map
.
时提供散列仿函数
您没有证明您已经进行了 std::hash<Edge>
专业化,并且您没有创建带有仿函数的 unordered_map
来进行散列,这就是默认构造失败的原因。
要添加 std::hash<Edge>
专业化,您可以这样做(如果您使 Edge
public):
namespace std {
template<>
struct hash<Graph::Edge> {
size_t operator()(const Graph::Edge& e) const {
size_t hash_result;
// calculate the hash result
return hash_result;
}
};
} // namespace std
如果您想保留 Edge
private
,使用仿函数可能更容易。这是一个 struct
形式的示例,它使 unordered_map
默认可构造:
class Graph {
private:
struct Edge {
// ...
};
// the hashing functor:
struct edge_hasher {
size_t operator()(const Edge& e) const {
// an example implementation using boost::hash_combine
// from <boost/container_hash/hash.hpp>:
std::hash<std::string> h;
size_t hash_result = 0;
boost::hash_combine(hash_result, h(e.vertex1));
boost::hash_combine(hash_result, h(e.vertex2));
boost::hash_combine(hash_result, std::hash<int>{}(e.val));
return hash_result;
}
};
std::unordered_map<Edge, Edge*, edge_hasher> edges; // <- hasher included
public:
Graph() = default;
~Graph();
}
我有以下图表 class:
class Graph {
private:
struct Edge {
string vertex1{}, vertex2{};
int val{};
Edge() = default;
~Edge() = default;
explicit Edge(string v1, string v2, int value) : vertex1(std::move(v1)), vertex2(std::move(v2)), val(value) {};
bool operator==(const Edge&) const;
};
unordered_map<Edge, Edge*> edges;
public:
Graph() = default;
~Graph();
}
当我想用默认构造函数构造一个图时,它说Explicitly defaulted default constructor of 'Graph' is implicitly deleted because field 'edges' has a deleted default constructor
。我应该如何更改我的代码,以便能够使用 Graph 的默认构造函数?
unordered_map
中的 Key 需要可哈希。通过添加 std::hash
class 模板的特化,或者在创建 unordered_map
.
您没有证明您已经进行了 std::hash<Edge>
专业化,并且您没有创建带有仿函数的 unordered_map
来进行散列,这就是默认构造失败的原因。
要添加 std::hash<Edge>
专业化,您可以这样做(如果您使 Edge
public):
namespace std {
template<>
struct hash<Graph::Edge> {
size_t operator()(const Graph::Edge& e) const {
size_t hash_result;
// calculate the hash result
return hash_result;
}
};
} // namespace std
如果您想保留 Edge
private
,使用仿函数可能更容易。这是一个 struct
形式的示例,它使 unordered_map
默认可构造:
class Graph {
private:
struct Edge {
// ...
};
// the hashing functor:
struct edge_hasher {
size_t operator()(const Edge& e) const {
// an example implementation using boost::hash_combine
// from <boost/container_hash/hash.hpp>:
std::hash<std::string> h;
size_t hash_result = 0;
boost::hash_combine(hash_result, h(e.vertex1));
boost::hash_combine(hash_result, h(e.vertex2));
boost::hash_combine(hash_result, std::hash<int>{}(e.val));
return hash_result;
}
};
std::unordered_map<Edge, Edge*, edge_hasher> edges; // <- hasher included
public:
Graph() = default;
~Graph();
}