提升变体复制语义
boost variant copy semantics
我想知道 boost 变体的复制语义是什么。我检查了源代码,这让我有点莫名其妙,所以我想知道,在示例代码中,我的 getVal(name)
函数在返回时是否复制了底层向量?如果是这样,我是否应该将其更改为返回的引用 (&)?
using Val = boost::variant<std::vector<int>, std::vector<std::string>>;
Val getVal(std::string& name) {
return map[name];// where map is std::map<std::string, Val>
}
是的,您的 getVal
returns 整个向量的 copy(包括所有元素字符串的副本,例如)。
是的,返回引用可以解决这个问题。
注意您还可以有一个存储引用的变体。在这种情况下,通过 "value" 返回它 仍然与返回引用具有相同的语义:
using Ref = variant<std::vector<int>&, std::vector<std::string>&>;
Ref getVal(std::string& name) {
return map[name]; // where map is std::map<std::string, Val>
}
具有从 Ref
转换为 Val
(反之亦然)的必要机制的完整样本:
#include <boost/variant.hpp>
#include <map>
#include <vector>
#include <string>
using Val = boost::variant<std::vector<int>, std::vector<std::string>>;
using Ref = boost::variant<std::vector<int>&, std::vector<std::string>& >;
std::map<std::string, Val> map {
{ "first", std::vector<int> { 1,2,3,4 } },
{ "2nd", std::vector<std::string> { "five", "six", "seven", "eight" } }
};
namespace { // detail
template <typename T>
struct implicit_convert : boost::static_visitor<T> {
template <typename U> T operator()(U&& u) const { return std::forward<U>(u); }
};
}
Ref getVal(std::string& name) {
return boost::apply_visitor(implicit_convert<Ref>(), map[name]);
}
#include <iostream>
int main() {
for (auto i : boost::get<std::vector<int> >(map["first"])) std::cout << i << " ";
for (auto i : boost::get<std::vector<std::string> >(map["2nd"])) std::cout << i << " ";
}
输出:
1 2 3 4 five six seven eight
没有任何向量被复制
我想知道 boost 变体的复制语义是什么。我检查了源代码,这让我有点莫名其妙,所以我想知道,在示例代码中,我的 getVal(name)
函数在返回时是否复制了底层向量?如果是这样,我是否应该将其更改为返回的引用 (&)?
using Val = boost::variant<std::vector<int>, std::vector<std::string>>;
Val getVal(std::string& name) {
return map[name];// where map is std::map<std::string, Val>
}
是的,您的 getVal
returns 整个向量的 copy(包括所有元素字符串的副本,例如)。
是的,返回引用可以解决这个问题。
注意您还可以有一个存储引用的变体。在这种情况下,通过 "value" 返回它 仍然与返回引用具有相同的语义:
using Ref = variant<std::vector<int>&, std::vector<std::string>&>;
Ref getVal(std::string& name) {
return map[name]; // where map is std::map<std::string, Val>
}
具有从 Ref
转换为 Val
(反之亦然)的必要机制的完整样本:
#include <boost/variant.hpp>
#include <map>
#include <vector>
#include <string>
using Val = boost::variant<std::vector<int>, std::vector<std::string>>;
using Ref = boost::variant<std::vector<int>&, std::vector<std::string>& >;
std::map<std::string, Val> map {
{ "first", std::vector<int> { 1,2,3,4 } },
{ "2nd", std::vector<std::string> { "five", "six", "seven", "eight" } }
};
namespace { // detail
template <typename T>
struct implicit_convert : boost::static_visitor<T> {
template <typename U> T operator()(U&& u) const { return std::forward<U>(u); }
};
}
Ref getVal(std::string& name) {
return boost::apply_visitor(implicit_convert<Ref>(), map[name]);
}
#include <iostream>
int main() {
for (auto i : boost::get<std::vector<int> >(map["first"])) std::cout << i << " ";
for (auto i : boost::get<std::vector<std::string> >(map["2nd"])) std::cout << i << " ";
}
输出:
1 2 3 4 five six seven eight
没有任何向量被复制