我一直在尝试将std::tuple
与参考文献结合使用:
#include <iostream>
#include <tuple>
int main() {
int a,b;
std::tuple<int&,int&> test(a,b);
std::get<0>(test) = 1;
std::get<1>(test) = 2;
std::cout << a << ":" << b << std::endl;
// doesn't make ref, not expected
auto test2 = std::make_tuple(a,b);
std::get<0>(test2) = -1;
std::get<1>(test2) = -2;
std::cout << a << ":" << b << std::endl;
int &ar=a;
int &br=b;
// why does this not make a tuple of int& references? can we force it to notice?
auto test3 = std::make_tuple(ar,br);
std::get<0>(test3) = -1;
std::get<1>(test3) = -2;
std::cout << a << ":" << b << std::endl;
}
在这三个例子中,前两个按预期工作。然而,第三个没有。我希望auto
类型(test3
)与test
的类型(即std::tuple<int&,int&>
)相同。
似乎std::make_tuple
无法自动生成引用元组。为什么不?除了明确地自己构建那种类型的东西之外,我能做些什么呢?
(编译器是g ++ 4.4.5,using 4.5 doesn't change it)
答案 0 :(得分:53)
std::tie
会进行非const
引用。
auto ref_tuple = std::tie(a,b); // decltype(ref_tuple) == std::tuple<int&, int&>
对于const
引用,您要么需要std::cref
包装函数:
auto cref_tuple = std::make_tuple(std::cref(a), std::cref(b));
或者使用简单的as_const
助手来限定变量,然后再将其传递给std::tie
:
template<class T>
T const& as_const(T& v){ return v; }
auto cref_tuple = std::tie(as_const(a), as_const(b));
或者,如果您想获得幻想,请编写自己的ctie
(重复使用std::tie
和as_const
):
template<class... Ts>
std::tuple<Ts const&...> ctie(Ts&... vs){
return std::tie(as_const(vs)...);
}
auto cref_tuple = ctie(a, b);
答案 1 :(得分:29)
尝试forward_as_tuple
:
auto test3 = std::forward_as_tuple(ar,br);
答案 2 :(得分:4)
怎么样:
auto test3 = std::make_tuple(std::ref(a),std::ref(b));
答案 3 :(得分:3)
为什么:make_tuple
参数由const引用(const T&
)传递,因此如果您传递int&
,T
匹配int
。如果它将T
推导为int&
,则参数将为const T&&
,并且您将收到编译错误。
答案 4 :(得分:-1)
在C ++ 14中,您可以这样继续:
template<typename ...T, size_t... I>
auto make_rtuple_helper(std::tuple<T...>& t , std::index_sequence<I...>)
-> std::tuple<T&...>
{ return std::tie(std::get<I>(t)...) ;}
template<typename ...T>
std::tuple<T&...> make_rtuple( std::tuple<T...>& t )
{
return make_rtuple_helper( t, std::make_index_sequence<sizeof...(T)>{});
}
请参阅coliru中的工作原理:http://coliru.stacked-crooked.com/a/a665130e17fd8bcc
干杯 A.A。
答案 5 :(得分:-2)
std::make_tuple<int&, int&>(a, b);
不可否认,它有点挫败了目的,但对于像make_shared
这样的功能,你仍然可以获益。
警告,我没有尝试编译它,但我相信它会起作用。