Получить индекс в С++ 11 foreach loop

Есть ли удобный способ получить индекс текущей записи контейнера в цикле foreach С++ 11, например enumerate в python:

for idx, obj in enumerate(container):
    pass

Я мог представить себе итератор, который также может вернуть индекс или аналогичный.

Конечно, я мог бы иметь счетчик, но часто итераторы не дают гарантий того, как они будут перебирать контейнер.

Ответ 1

Хорошую реализацию запрашиваемой функции можно найти здесь:

https://github.com/ignatz/pythonic

Идея заключается в том, что вы создаете структуру-оболочку с пользовательским итератором, который выполняет подсчет. Ниже приведена минимальная иллюстративная реализация, иллюстрирующая идею:

#include <iostream>
#include <vector>
#include <tuple>

// Wrapper class
template <typename T>
class enumerate_impl
{
public:
    // The return value of the operator* of the iterator, this
    // is what you will get inside of the for loop
    struct item
    {
        size_t index;
        typename T::value_type & item;
    };
    typedef item value_type;

    // Custom iterator with minimal interface
    struct iterator
    {
        iterator(typename T::iterator _it, size_t counter=0) :
            it(_it), counter(counter)
        {}

        iterator operator++()
        {
            return iterator(++it, ++counter);
        }

        bool operator!=(iterator other)
        {
            return it != other.it;
        }

        typename T::iterator::value_type item()
        {
            return *it;
        }

        value_type operator*()
        {
            return value_type{counter, *it};
        }

        size_t index()
        {
            return counter;
        }

    private:
        typename T::iterator it;
        size_t counter;
    };

    enumerate_impl(T & t) : container(t) {}

    iterator begin()
    {
        return iterator(container.begin());
    }

    iterator end()
    {
        return iterator(container.end());
    }

private:
    T & container;
};

// A templated free function allows you to create the wrapper class
// conveniently 
template <typename T>
enumerate_impl<T> enumerate(T & t)
{
    return enumerate_impl<T>(t);
}



int main()
{
    std::vector<int> data = {523, 1, 3};
    for (auto x : enumerate(data))
    {
        std::cout << x.index << ": " << x.item << std::endl;
    }
}

Ответ 2

Как насчет простого решения, такого как:

int counter=0;
for (auto &val: container)
{
    makeStuff(val, counter);

    counter++;
}

Вы можете немного усложнить добавление кода после счетчика, добавив область действия:

int counter=0;
for (auto &val: container)
{{
    makeStuff(val, counter); 
}counter++;}

Как указал @graham.reeds, нормальный цикл for также является решением, которое может быть таким же быстрым:

int counter=0;
for (auto it=container.begin(); it!=container.end(); ++it, ++counter)
{
    makeStuff(val, counter);
}

И, наконец, альтернативный способ использования алгоритма:

int counter = 0;
std::for_each(container.begin(), container.end(), [&counter](int &val){ 
    makeStuff(val, counter++);
});

Примечание: порядок между циклом диапазона и нормальным контуром гарантируется стандартом 6.5.4. Значение счетчика может быть согласованным с положением в контейнере.

Ответ 3

Если у вас есть доступ к Boost it, адаптеры диапазона можно использовать следующим образом:

using namespace boost::adaptors;

for (auto const& elem : container | indexed(0))
{
    std::cout << elem.index() << " - " << elem.value() << '\n';
}

Источник (где есть и другие примеры)

Ответ 4

Если вам нужен индекс, тогда традиционный для работы отлично работает.

for (int idx=0; idx<num; ++idx)
{
// do stuff
}