%PDF- %PDF-
Mini Shell

Mini Shell

Direktori : /proc/985914/root/lib/python3/dist-packages/pythran/pythonic/itertools/
Upload File :
Create Path :
Current File : //proc/985914/root/lib/python3/dist-packages/pythran/pythonic/itertools/permutations.hpp

#ifndef PYTHONIC_ITERTOOLS_PERMUTATIONS_HPP
#define PYTHONIC_ITERTOOLS_PERMUTATIONS_HPP

#include "pythonic/include/itertools/permutations.hpp"
#include "pythonic/utils/functor.hpp"
#include "pythonic/types/dynamic_tuple.hpp"
#include "pythonic/builtins/range.hpp"

#include <algorithm>

PYTHONIC_NS_BEGIN

namespace itertools
{

  template <class T>
  permutations_iterator<T>::permutations_iterator()
  {
  }

  template <class T>
  permutations_iterator<T>::permutations_iterator(
      std::vector<typename T::value_type> const &iter, size_t num_elts,
      bool end)
      : pool(iter), curr_permut(pool.size()), _size(num_elts), end(end)
  {
    std::iota(curr_permut.begin(), curr_permut.end(), 0);
    if (num_elts > iter.size()) {
      end = true;
    }
  }

  template <class T>
  types::dynamic_tuple<typename T::value_type> permutations_iterator<T>::
  operator*() const
  {
    std::vector<typename T::value_type> res(_size);
    for (size_t i = 0; i < _size; i++)
      res[i] = pool[curr_permut[i]]; // Ok because types::dynamic_tuple is
                                     // indeed a vector
    return {res.begin(), res.end()};
  }

  template <class T>
  permutations_iterator<T> &permutations_iterator<T>::operator++()
  {
    if (_size != pool.size()) {
      // Slow path, the iterator is a "view" of a prefix smaller
      // than the the pool size
      // FIXME a better implementation would be to avoid
      // std::next_permutation, but only in the slow path
      types::dynamic_tuple<long> prev_permut(curr_permut.begin(),
                                             curr_permut.begin() + _size);
      while ((end = std::next_permutation(curr_permut.begin(),
                                          curr_permut.end()))) {
        // Check if the prefix of the new permutation is
        // different of the previous one
        types::dynamic_tuple<long> new_permut(curr_permut.begin(),
                                              curr_permut.begin() + _size);
        if (!(prev_permut == new_permut))
          break;
      }
    } else
      end = std::next_permutation(curr_permut.begin(), curr_permut.end());
    return *this;
  }

  template <class T>
  bool permutations_iterator<T>::
  operator!=(permutations_iterator<T> const &other) const
  {
    return !(*this == other);
  }

  template <class T>
  bool permutations_iterator<T>::
  operator==(permutations_iterator<T> const &other) const
  {
    if (other.end != end)
      return false;
    return std::equal(curr_permut.begin(), curr_permut.end(),
                      other.curr_permut.begin());
  }

  template <class T>
  bool permutations_iterator<T>::
  operator<(permutations_iterator<T> const &other) const
  {
    if (end != other.end)
      return end > other.end;
    for (long i = 0; i < pool.size(); i++)
      if (other.curr_permut[i] < curr_permut[i])
        return false;
      else if (other.curr_permut[i] > curr_permut[i])
        return true;
    return false;
  }

  template <class T>
  _permutations<T>::_permutations()
  {
  }

  template <class T>
  _permutations<T>::_permutations(T iter, long elts)
      : iterator(std::vector<typename T::value_type>(iter.begin(), iter.end()),
                 elts, true)
  {
  }

  template <class T>
  typename _permutations<T>::iterator const &_permutations<T>::begin() const
  {
    return *this;
  }

  template <class T>
  typename _permutations<T>::iterator _permutations<T>::begin()
  {
    return *this;
  }

  template <class T>
  typename _permutations<T>::iterator _permutations<T>::end() const
  {
    return iterator(iterator::pool, iterator::_size, false);
  }

  template <typename T0>
  _permutations<T0> permutations(T0 iter, long num_elts)
  {
    return _permutations<T0>(iter, num_elts);
  }

  template <typename T0>
  _permutations<T0> permutations(T0 iter)
  {
    return _permutations<T0>(iter, std::distance(iter.begin(), iter.end()));
  }
}
PYTHONIC_NS_END

#endif

Zerion Mini Shell 1.0