2013-10-10 94 views
1

这不是秘密,std::get<i>(tuple)让很多程序员很烦恼。通过索引C++访问元组元素11

取而代之,我想用tuple[i]之类的东西。

所以我试图模拟它。

#include <iostream> 
#include <type_traits> 
#include <tuple> 

template <int> struct index{}; 



template< char ... > 
struct combine; 

template<> struct combine<> : std::integral_constant< int , 0>{}; 


constexpr int ten(size_t p)noexcept 
{ 
    return p == 0 ? 1 : 10 * ten(p-1); 
} 

template< char c, char ... t> 
struct combine<c, t...> : std::integral_constant< int, (c - '0')*ten(sizeof...(t)) + combine<t...>::value > 
{ static_assert(c >= '0' && c <= '9', "only 0..9 digits are allowed"); }; 


template< char ... c > 
constexpr auto operator "" _index()noexcept 
{ 
    return index< combine<c...>::value >{}; 
}; 

template< class ... Args > 
struct mytuple : public std::tuple<Args...> 
{ 
    using std::tuple<Args...>::tuple; 

    template< int i > 
    auto& operator [](index<i>) noexcept 
    { 
     return std::get<i> (static_cast< std::tuple<Args...> & >(*this)); 
    } 

    template< int i> 
    auto const& operator [](index<i>)const noexcept 
    { 
     return std::get<i>(static_cast< std::tuple<Args...> const& >(*this)); 
    } 

}; 


int main() 
{ 
    static_assert(combine<'1','2','3','4'>::value == 1234, "!"); 


    static_assert(std::is_same< index<785>, decltype(785_index) > {}, "!"); 

    using person = mytuple< std::string, int, double, char>; 

    person s = std::make_tuple("Bjarne Stroustrup", 63, 3.14, '7'); 
    auto name = s[ 0_index ]; 
    auto old = s[ 1_index ]; 
    auto number = s[ 2_index ]; 
    auto symbol = s[ 3_index ]; 

    std::cout << "name: " << name << '\t' 
       << "old: " << old << '\t' 
       << "number: " << number<< '\t' 
       << "symbol: " << symbol<< '\t' 
       << std::endl; 
} 

问:这段代码有什么问题?即此代码是否可用? 如果可以使用为什么不这样执行std::tuple

+4

海事组织很少有这么多的工作。我也认为这个例子有一个错误。尝试'0x1_index'或甚至'1.0_index'。 – Simple

+0

@简单。谢谢。 –

+0

我也会说这不值得。 'std :: get (tuple)'这个令人讨厌的事情是它不能与变量一起使用,而不能通过索引操作符来完成。您的实现仍然不提供通过简单变量访问元组。 –

回答

5

我不确定你的案例中的具体问题是什么,但看起来你正在寻找一点便利。下面以占位符(开始_1),以简化代码:

#include <iostream> 
#include <type_traits> 
#include <tuple> 
#include <functional> 

template< class ... Args > 
struct mytuple : public std::tuple<Args...> 
{ 
    using std::tuple<Args...>::tuple; 

    template< typename T > 
    auto& operator [](T) noexcept 
    { 
     return std::get< std::is_placeholder<T>::value - 1 >(*this); 
    } 

    template< typename T > 
    auto const& operator [](T) const noexcept 
    { 
     return std::get< std::is_placeholder<T>::value - 1 >(*this); 
    } 
}; 

int main() 
{ 
    using person = mytuple< std::string, int, double, char>; 

    using namespace std::placeholders; 

    person s = std::make_tuple("Bjarne Stroustrup", 63, 3.14, '7'); 
    auto name = s[ _1 ]; 
    auto old = s[ _2 ]; 
    auto number = s[ _3 ]; 
    auto symbol = s[ _4 ]; 

    std::cout << "name: " << name << '\t' 
       << "old: " << old << '\t' 
       << "number: " << number<< '\t' 
       << "symbol: " << symbol<< '\t' 
       << std::endl; 
} 

这里的关键是要知道std::is_placeholder<T>保证从std::integral_constant<int,N>衍生:)希望你喜欢它。

+2

可惜它不是基于零的。 –

+0

在这种情况下'static_cast'是不必要的。 – Xeo

+0

这很好,但1)它不是基于零的。 2)最大_N - 由编译器定义的实现。我并不感兴趣如何实现这一点,我很有趣,为什么这种方式不流行或不被使用? –