Outils pour utilisateurs

Outils du site


lang:cpp:boucles

Ceci est une ancienne révision du document !


La boucle traditionnelle

Intervalle

for (int i = 0; i < 100; i++)
  std::cout << i << std::endl;

Chaque paramètre est optionnel :

int i = 0;
for (;;)
{
  if (i >= 100)
    break;
  std::cout << i << std::endl;
  i++;
}

Et sa version while :

int i = 0;
while (i < 100)
{
  std::cout << i << std::endl;
  i++;
}

Conteneur supportant les index

std::array<int, 4> arr { 10, 20, 30, 40 };
for (size_t i = 0; i < arr.size(); i++)
  std::cout << arr[i] << std::endl;

Conteneur supportant les itérateurs

std::vector<int> vector{ 10, 20, 30, 40 };
std::vector<int>::const_iterator it;
std::vector<int>::iterator itc;
 
// Lecture / écriture
for (itc = vector.begin(); itc != vector.end(); itc++)
  (*itc)++;
 
// Lecture seule
for (it = vector.cbegin(); it != vector.cend(); it++)
  std::cout << *it << std::endl;

for each

Conteneur de type primitif

std::array<int, 4> arr = { 10, 20, 30, 40 };
for (int i : arr)
  std::cout << i << std::endl;

Conteneur de type non primitif

std::set<std::string> set { "10", "20", "30", "40" };
 
for (const std::string& val : set)
  std::cout << val << std::endl;

Conteneur multi-types (''std::map'', ''std::tuple'', ...)

std::map<int, long> map { {10, 10}, {20, 20}, {30, 30}, {40, 40} };
 
// L'utilisation de auto est obligatoire.
for (const auto & [key, value] : map)
  std::cout << value << std::endl;

L'interprétation par le compilateur sera :

std::map<int, long> map = std::map<int, long, std::less<int>, std::allocator<std::pair<const int, long> > >{std::initializer_list<std::pair<const int, long> >{std::pair<const int, long>{10, 10}, std::pair<const int, long>{20, 20}, std::pair<const int, long>{30, 30}, std::pair<const int, long>{40, 40}}, std::less<int>(), std::allocator<std::pair<const int, long> >()};
{
  std::map<int, long, std::less<int>, std::allocator<std::pair<const int, long> > > & __range1 = map;
  std::_Rb_tree_iterator<std::pair<const int, long> > __begin1 = __range1.begin();
  std::_Rb_tree_iterator<std::pair<const int, long> > __end1 = __range1.end();
  for(; __begin1.operator!=(__end1); __begin1.operator++()) 
  {
    const std::pair<const int, long> & __operator9 = __begin1.operator*();
    std::tuple_element<0, const std::pair<const int, long> >::type& key = std::get<0UL>(__operator9);
    std::tuple_element<1, const std::pair<const int, long> >::type& value = std::get<1UL>(__operator9);
    std::cout.operator<<(value).operator<<(std::endl);
  }
}

Sur une classe personnalisée

Il faut définir l'itérateur et la classe à parcourir. C++11 range-based for loops Archive le 26/12/2019

L'exemple ci-dessous se passe de template.

Il faut commencer par déclarer l'itérateur.

#include <iostream>
 
class DataSample;
 
// Données accessibles depuis la boucle.
class Data
{
 public:
  int getA() const { return a; }
  void setA(int aa) { a = aa; }
 
 private:
  int a;
};
 
// Itérateur accessible en écriture.
class Iter
{
 public:
  Iter(DataSample &p_vec, int pos) : _pos(pos), _p_vec(p_vec) {}
 
  bool operator!=(const Iter &other) const { return _pos != other._pos; }
  Data &operator*();
  Iter &operator++()
  {
    ++_pos;
    return *this;
  }
 
 private:
  int _pos;
  DataSample &_p_vec;
};
 
// Iterateur accessible uniquement en lecture.
class ConstIter
{
 public:
  ConstIter(const DataSample &p_vec, int pos) : _pos(pos), _p_vec(p_vec) {}
 
  bool operator!=(const ConstIter &other) const { return _pos != other._pos; }
  const Data &operator*() const;
  const ConstIter &operator++()
  {
    ++_pos;
    return *this;
  }
 
 private:
  int _pos;
  const DataSample &_p_vec;
};
 
// Classe stockant les données et implémentant le pattern for each.
class DataSample
{
 public:
  // Méthodes accessibles en écriture.
  Data &get(int col) { return _data[col]; }
  Iter begin() { return Iter(*this, 0); }
  Iter end() { return Iter(*this, 100); }
  // Méthodes accessibles uniquement en lecture seule.
  const Data &get(int col) const { return _data[col]; }
  ConstIter begin() const { return ConstIter(*this, 0); }
  ConstIter end() const { return ConstIter(*this, 100); }
 
  void set(int index, int val) { _data[index].setA(val); }
 
 private:
  // Dans cette classe, les données sont sous forme d'un tableau.
  Data _data[100];
};
 
Data &Iter::operator*() { return _p_vec.get(_pos); }
const Data &ConstIter::operator*() const { return _p_vec.get(_pos); }
 
// Exemple d'usage.
int main()
{
  DataSample v;
  for (int i = 0; i < 100; i++)
  {
    v.set(i, i);
  }
  // Utilisation de begin et end en non const.
  // Le type de retour est Data& (qui peut être casté en const Data&).
  for (Data &i : v)
  {
    std::cout << i.getA() << std::endl;
  }
  // Utilisation de begin et end en const.
  // Le type de retour est obligatoirement const Data&.
  for (const Data &i : static_cast<const DataSample>(v))
  {
    std::cout << i.getA() << std::endl;
  }
}

Via une coroutine

#include <experimental/generator>
#include <iostream>
 
std::experimental::generator<int> loop(int iterations)
{
  for (int i = 0; i < iterations; i++)
  {
    co_yield i;
  }
}
 
int main()
{
  for (int i : loop(100))
  {
    std::cout << i << std::endl;
  }
}
lang/cpp/boucles.1582033899.txt.gz · Dernière modification : 2020/02/18 14:51 de root