C++11范围for循环

范围for循环:

1.基于范围的for循环

for(元素类型 元素对象:容器对象)

{

  循环体

}

(1.1)如果循环体由单条语句或者单个结构块组成,可以省略花括号

(1.2)用元素对象依次结合容器对象中的每一个元素,每结合一个元素,执行依次循环体,直至容器内的所有元素都被结合完为止.

(1.3)不依赖于下标元素,通用

(1.4)不需要访问迭代器,透明

(1.5)不需要定义处理函数,简洁

#include "stdafx.h"
#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
void print(int i)
{
    cout << i << " ";
}

int main()
{
    int ai[]{ 65, 66, 67, 68, 69 };
    //计算数组元素个数
    size_t size = sizeof(ai) / sizeof(ai[0]);
    vector<int> vi(ai, ai + size);
    //基于下标运算的for循环,不是所有的容器都支持下标运算,不通用
    for (size_t i = 0; i < size; ++i)
        cout << ai[i] << " ";
    cout << endl;
    for (size_t i = 0; i < vi.size(); ++i)
        cout << vi[i] << " ";
    cout << endl;
    //基于迭代器的for循环,需要指明容器两端,且对迭代器做自增,必须了解迭代器的运算规则,不够透明
    for (int *it = ai; it != ai + size; ++it)
        cout << *it << " ";
    cout << endl;
    for (auto it = vi.begin(); it != vi.end(); ++it)
        cout << *it << " ";
    cout << endl;
    //基于泛型函数的for循环,需要提供针对元素的处理函数,对于一般性遍历而言比较繁琐
    for_each(ai, ai + size, print);
    cout << endl;
    for_each(vi.begin(), vi.end(), print);
    cout << endl;
    for (auto a : ai)
        cout << a << " ";
    cout << endl;
    for (auto a : vi)
        cout << a << " ";
    cout << endl;
    for (auto &a : ai)
        ++a;
    for (auto &a : vi)
        --a;
    for (auto const &a : ai)
        cout << a << " ";
    cout << endl;
    for (auto const &a : vi)
        cout << a << " ";
    cout << endl;
    for (char a : ai)
        cout << a << " ";
    cout << endl;
    for (char a : vi)
        cout << a << " ";
    cout << endl;
    return 0;
}

2.范围循环的注意事项

(2.1)对map和multimap容器使用范围循环,每次拿到的元素既不是键也不是值,而是由键和值组成的pair

(2.2)在使用基于范围的for循环时,不能违背容器本身的约束

(2.3)基于范围的for循环,无论循环体执行多少次,冒号后面的表达式永远只执行一次

(2.4)基于范围的for循环,其底层实现依然要借助于容器的迭代器,

因此任何可能导致迭代器失效的结构性改变,都可能引发未定义的后果

#include <iostream>
#include <string>
#include <map>
#include <list>
using namespace std;

list<int> getScores(void)
{
    cout << __FUNCTION__ << endl;
    return{ 70, 75, 80, 85, 90, 95 };
}

int main()
{
    //对map和multimap容器使用范围循环,每次拿到的元素既不是键也不是值,而是由键和值组成的pair
    multimap<string, int> msi;
    msi.insert(make_pair("张飞", 100));
    msi.insert(make_pair("赵云", 90));
    msi.insert(make_pair("关羽", 80));
    for (auto c : msi)
        cout << c.first << ":" << c.second << endl;
    cout << endl;
    for (auto it = msi.begin(); it != msi.end(); ++it)
        cout << it->first << ":" << it->second << endl;
    cout << endl;

    //在使用基于范围的for循环时,不能违背容器本身的约束
    /*for (pair<string,int> &c:msi)
    if (c.first == "张飞")
    c.first = "张菲菲";*/
    for (auto c : msi)
        cout << c.first << ":" << c.second << endl;
    cout << endl;
    //基于范围的for循环,无论循环体执行多少次,冒号后面的表达式永远只执行一次
    for (auto score : getScores())
        cout << score << " ";
    cout << endl;
    auto scores = getScores();
    for (auto score : scores)
    {
        cout << score << " ";
        //基于范围的for循环,其底层实现依然要借助于容器的迭代器,
        //因此任何可能导致迭代器失效的结构性改变,都可能引发未定义的后果
        //scores.pop_front();
    }
    cout << endl;
    return 0;
}

3.使自己定义的容器类型支持范围循环

一个类只要提供了分别获取起始和终止迭代器的begin和end函数,就可以支持基于范围的for循环

#include "stdafx.h"
#include <iostream>
using namespace std;

template <typename T, size_t S>
class Array
{
public:
    T &operator[](size_t i)
    {
        return m_array[i];
    }
    T const &operator[](size_t i)const
    {
        return const_cast<Array&>(*this)[i];
    }
    //获取起始迭代器
    T *begin()
    {
        return m_array;
    }
    T const *begin()const
    {
        return const_cast<Array *>(this)->begin();
    }
    //获取终止迭代器
    T *end()
    {
        return m_array + S;
    }
    T const *end()const
    {
        return const_cast<Array *>(this)->end();
    }
private:
    T m_array[S];
};


int main()
{
    int i = 0;
    Array<int, 5> ai;
    for (auto &a : ai)
        a = ++i * 10;
    auto const &cai = ai;
    for (auto &a : cai)
        cout << /*++*/a << " ";
    cout << endl;
    return 0;
}