选择"常用功能"迭代器类别

时间:2018-04-10 14:56:34

标签: c++

当您处理不同类型的迭代器并想知道所有操作都可用时,有很多情况。我希望你能从它们的迭代器类别中推断出它。

例如,让我们说你正在撰写chain_iteratorzip_iterator,并希望从其组成部分定义iterator_category' iterator_category秒。这是一种有效的方法:

  1. 如果至少有一个是output_iterator,则公共类别为output_iterator_tag,除非其中至少有一个是input_iterator,在这种情况下,公共类别未定义

  2. 如果至少有一个是intput_iterator,则公共类别为input_iterator_tag,除非其中至少有一个是output_iterator,在这种情况下,公共类别未定义

  3. 否则公共类别被推断为forward_iterator_tagbidirectional_iterator_tagrandom_access_iterator_tag中的第一个出现在基础迭代器列表中的第一个类别'类型。

  4. 在代码中说出来:

    template <typename... Iterators>
    using least_powerful_iterator =
        std::conditional_t<contains<std::input_iterator_tag,         category<Iterators>...> &&
                          !contains<std::output_iterator_tag,        category<Iterators>...>, std::input_iterator_tag,
        std::conditional_t<contains<std::input_iterator_tag,         category<Iterators>...> &&
                          contains<std::output_iterator_tag,         category<Iterators>...>, unknown_iterator_category,
        std::conditional_t<contains<std::output_iterator_tag,        category<Iterators>...> &&
                          !contains<std::input_iterator_tag,         category<Iterators>...>, std::output_iterator_tag,
        std::conditional_t<contains<std::output_iterator_tag,        category<Iterators>...> &&
                          contains<std::input_iterator_tag,          category<Iterators>...>, unknown_iterator_category,    
        std::conditional_t<contains<std::forward_iterator_tag,       category<Iterators>...>, std::forward_iterator_tag,
        std::conditional_t<contains<std::bidirectional_iterator_tag, category<Iterators>...>, std::bidirectional_iterator_tag,
        std::conditional_t<contains<std::random_access_iterator_tag, category<Iterators>...>, std::random_access_iterator_tag,
        unknown_iterator_category>>>>>>>;
    

1 个答案:

答案 0 :(得分:3)

std::common_type应该有效,不是吗?迭代器特征具有派生关系:

#include <iostream>
#include <type_traits>
#include <typeinfo>
#include <map>
#include <forward_list>
#include <iterator>

int main()
{
    using map_iter = std::map<int, int>::iterator;
    using fl_iter = std::forward_list<double>::iterator;

    using cat = std::common_type_t<
        std::iterator_traits<map_iter>::iterator_category, 
        std::iterator_traits<fl_iter>::iterator_category>;

    std::cout << typeid(cat).name() << std::endl;
}