diff --git a/src/common/aka_array.hh b/src/common/aka_array.hh
index 46300822e..a2523289d 100644
--- a/src/common/aka_array.hh
+++ b/src/common/aka_array.hh
@@ -1,399 +1,363 @@
 /**
  * @file   aka_array.hh
  *
  * @author Till Junge <till.junge@epfl.ch>
  * @author Nicolas Richart <nicolas.richart@epfl.ch>
  *
  * @date creation: Fri Jun 18 2010
  * @date last modification: Fri Jan 22 2016
  *
  * @brief  Array container for Akantu
  * This container differs from the std::vector from the fact it as 2 dimensions
  * a main dimension and the size stored per entries
  *
  * @section LICENSE
  *
  * Copyright (©)  2010-2012, 2014,  2015 EPFL  (Ecole Polytechnique  Fédérale de
  * Lausanne)  Laboratory (LSMS  -  Laboratoire de  Simulation  en Mécanique  des
  * Solides)
  *
  * Akantu is free  software: you can redistribute it and/or  modify it under the
  * terms  of the  GNU Lesser  General Public  License as  published by  the Free
  * Software Foundation, either version 3 of the License, or (at your option) any
  * later version.
  *
  * Akantu is  distributed in the  hope that it  will be useful, but  WITHOUT ANY
  * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
  * A  PARTICULAR PURPOSE. See  the GNU  Lesser General  Public License  for more
  * details.
  *
  * You should  have received  a copy  of the GNU  Lesser General  Public License
  * along with Akantu. If not, see <http://www.gnu.org/licenses/>.
  *
  */
 
 /* -------------------------------------------------------------------------- */
 
 #ifndef __AKANTU_VECTOR_HH__
 #define __AKANTU_VECTOR_HH__
 
 /* -------------------------------------------------------------------------- */
 #include "aka_common.hh"
 /* -------------------------------------------------------------------------- */
 #include <typeinfo>
 #include <vector>
 /* -------------------------------------------------------------------------- */
 
 namespace akantu {
 
 /// class that afford to store vectors in static memory
 class ArrayBase {
   /* ------------------------------------------------------------------------ */
   /* Constructors/Destructors                                                 */
   /* ------------------------------------------------------------------------ */
 public:
   explicit ArrayBase(ID id = "");
 
   virtual ~ArrayBase();
 
   /* ------------------------------------------------------------------------ */
   /* Methods                                                                  */
   /* ------------------------------------------------------------------------ */
 public:
   /// get the amount of space allocated in bytes
   inline UInt getMemorySize() const;
 
   /// set the size to zero without freeing the allocated space
   inline void empty();
 
   /// function to print the containt of the class
   virtual void printself(std::ostream & stream, int indent = 0) const;
 
   /* ------------------------------------------------------------------------ */
   /* Accessors */
   /* ------------------------------------------------------------------------ */
 public:
   /// Get the real size allocated in memory
   AKANTU_GET_MACRO(AllocatedSize, allocated_size, UInt);
   /// Get the Size of the Array
   UInt getsize() const __attribute__((deprecated)) { return size_; }
   UInt size() const { return size_; }
   /// Get the number of components
   AKANTU_GET_MACRO(NbComponent, nb_component, UInt);
   /// Get the name of th array
   AKANTU_GET_MACRO(ID, id, const ID &);
   /// Set the name of th array
   AKANTU_SET_MACRO(ID, id, const ID &);
 
   // AKANTU_GET_MACRO(Tag, tag, const std::string &);
   // AKANTU_SET_MACRO(Tag, tag, const std::string &);
 
   /* ------------------------------------------------------------------------ */
   /* Class Members                                                            */
   /* ------------------------------------------------------------------------ */
 protected:
   /// id of the vector
   ID id;
 
   /// the size allocated
   UInt allocated_size{0};
 
   /// the size used
   UInt size_{0};
 
   /// number of components
   UInt nb_component{1};
 
   /// size of the stored type
   UInt size_of_type{0};
 };
 
 /* -------------------------------------------------------------------------- */
+namespace {
+  template <std::size_t dim, typename T> struct IteratorHelper {};
+
+  template <typename T> struct IteratorHelper<0, T> { using type = T; };
+  template <typename T> struct IteratorHelper<1, T> { using type = Vector<T>; };
+  template <typename T> struct IteratorHelper<2, T> { using type = Matrix<T>; };
+  template <typename T> struct IteratorHelper<3, T> {
+    using type = Tensor3<T>;
+  };
+
+  template <std::size_t dim, typename T>
+  using IteratorHelper_t = typename IteratorHelper<dim, T>::type;
+} // namespace
 
 /* -------------------------------------------------------------------------- */
 template <typename T, bool is_scal> class Array : public ArrayBase {
   /* ------------------------------------------------------------------------ */
   /* Constructors/Destructors                                                 */
   /* ------------------------------------------------------------------------ */
 public:
   using value_type = T;
   using reference = value_type &;
   using pointer_type = value_type *;
   using const_reference = const value_type &;
 
   /// Allocation of a new vector
   explicit inline Array(UInt size = 0, UInt nb_component = 1,
                         const ID & id = "");
 
   /// Allocation of a new vector with a default value
   Array(UInt size, UInt nb_component, const value_type def_values[],
         const ID & id = "");
 
   /// Allocation of a new vector with a default value
   Array(UInt size, UInt nb_component, const_reference value,
         const ID & id = "");
 
   /// Copy constructor (deep copy if deep=true)
   Array(const Array<value_type, is_scal> & vect, bool deep = true,
         const ID & id = "");
 
 #ifndef SWIG
   /// Copy constructor (deep copy)
   explicit Array(const std::vector<value_type> & vect);
 #endif
 
   inline ~Array() override;
 
   Array & operator=(const Array & a) {
     /// this is to let STL allocate and copy arrays in the case of
     /// std::vector::resize
     AKANTU_DEBUG_ASSERT(this->size == 0, "Cannot copy akantu::Array");
     return const_cast<Array &>(a);
   }
 
   /* ------------------------------------------------------------------------ */
   /* Iterator                                                                 */
   /* ------------------------------------------------------------------------ */
   /// \todo protected: does not compile with intel  check why
 public:
   template <class R, class IR = R, bool issame = is_same<IR, T>::value>
   class iterator_internal;
 
 public:
   /* ------------------------------------------------------------------------ */
 
   /* ------------------------------------------------------------------------ */
   template <typename R = T> class const_iterator;
   template <typename R = T> class iterator;
 
   /* ------------------------------------------------------------------------ */
 
   /// iterator for Array of nb_component = 1
   using scalar_iterator = iterator<T>;
   /// const_iterator for Array of nb_component = 1
   using const_scalar_iterator = const_iterator<T>;
 
-  /// iterator rerturning Vectors of size n  on entries of Array with
+  /// iterator returning Vectors of size n  on entries of Array with
   /// nb_component = n
   using vector_iterator = iterator<Vector<T>>;
-  /// const_iterator rerturning Vectors of n size on entries of Array with
+  /// const_iterator returning Vectors of n size on entries of Array with
   /// nb_component = n
   using const_vector_iterator = const_iterator<Vector<T>>;
 
-  /// iterator rerturning Matrices of size (m, n) on entries of Array with
+  /// iterator returning Matrices of size (m, n) on entries of Array with
   /// nb_component = m*n
   using matrix_iterator = iterator<Matrix<T>>;
-  /// const iterator rerturning Matrices of size (m, n) on entries of Array with
+  /// const iterator returning Matrices of size (m, n) on entries of Array with
   /// nb_component = m*n
   using const_matrix_iterator = const_iterator<Matrix<T>>;
 
-  /* ------------------------------------------------------------------------ */
-  /// Get an iterator that behaves like a pointer T * to the first entry
-  inline scalar_iterator begin();
-  /// Get an iterator that behaves like a pointer T * to the end of the Array
-  inline scalar_iterator end();
-  /// Get a const_iterator to the beginging of an Array of scalar
-  inline const_scalar_iterator begin() const;
-  /// Get a const_iterator to the end of an Array of scalar
-  inline const_scalar_iterator end() const;
-
-  /// Get a scalar_iterator on the beginning of the Array considered of shape
-  /// (new_size)
-  inline scalar_iterator begin_reinterpret(UInt new_size);
-  /// Get a scalar_iterator on the end of the Array considered of shape
-  /// (new_size)
-  inline scalar_iterator end_reinterpret(UInt new_size);
-  /// Get a const_scalar_iterator on the beginning of the Array considered of
-  /// shape (new_size)
-  inline const_scalar_iterator begin_reinterpret(UInt new_size) const;
-  /// Get a const_scalar_iterator on the end of the Array considered of shape
-  /// (new_size)
-  inline const_scalar_iterator end_reinterpret(UInt new_size) const;
+  /// iterator returning Tensor3 of size (m, n, k) on entries of Array with
+  /// nb_component = m*n*k
+  using tensor3_iterator = iterator<Tensor3<T>>;
+  /// const iterator returning Tensor3 of size (m, n, k) on entries of Array
+  /// with nb_component = m*n*k
+  using const_tensor3_iterator = const_iterator<Tensor3<T>>;
 
   /* ------------------------------------------------------------------------ */
-  /// Get a vector_iterator on the beginning of the Array
-  inline vector_iterator begin(UInt n);
-  /// Get a vector_iterator on the end of the Array
-  inline vector_iterator end(UInt n);
-  /// Get a vector_iterator on the beginning of the Array
-  inline const_vector_iterator begin(UInt n) const;
-  /// Get a vector_iterator on the end of the Array
-  inline const_vector_iterator end(UInt n) const;
-
-  /// Get a vector_iterator on the begining of the Array considered of shape
-  /// (new_size, n)
-  inline vector_iterator begin_reinterpret(UInt n, UInt new_size);
-  /// Get a vector_iterator on the end of the Array considered of shape
-  /// (new_size, n)
-  inline vector_iterator end_reinterpret(UInt n, UInt new_size);
-  /// Get a const_vector_iterator on the begining of the Array considered of
-  /// shape (new_size, n)
-  inline const_vector_iterator begin_reinterpret(UInt n, UInt new_size) const;
-  /// Get a const_vector_iterator on the end of the Array considered of shape
-  /// (new_size, n)
-  inline const_vector_iterator end_reinterpret(UInt n, UInt new_size) const;
+  template <typename... Ns> inline decltype(auto) begin(Ns... n);
+  template <typename... Ns> inline decltype(auto) end(Ns... n);
 
-  /* ------------------------------------------------------------------------ */
-  /// Get a matrix_iterator on the begining of the Array (Matrices of size (m,
-  /// n))
-  inline matrix_iterator begin(UInt m, UInt n);
-  /// Get a matrix_iterator on the end of the Array (Matrices of size (m, n))
-  inline matrix_iterator end(UInt m, UInt n);
-  /// Get a const_matrix_iterator on the begining of the Array (Matrices of size
-  /// (m, n))
-  inline const_matrix_iterator begin(UInt m, UInt n) const;
-  /// Get a const_matrix_iterator on the end of the Array (Matrices of size (m,
-  /// n))
-  inline const_matrix_iterator end(UInt m, UInt n) const;
-
-  /// Get a matrix_iterator on the begining of the Array considered of shape
-  /// (new_size, m*n)
-  inline matrix_iterator begin_reinterpret(UInt m, UInt n, UInt size);
-  /// Get a matrix_iterator on the end of the Array considered of shape
-  /// (new_size, m*n)
-  inline matrix_iterator end_reinterpret(UInt m, UInt n, UInt size);
-  /// Get a const_matrix_iterator on the begining of the Array considered of
-  /// shape (new_size, m*n)
-  inline const_matrix_iterator begin_reinterpret(UInt m, UInt n,
-                                                 UInt size) const;
-  /// Get a const_matrix_iterator on the end of the Array considered of shape
-  /// (new_size, m*n)
-  inline const_matrix_iterator end_reinterpret(UInt m, UInt n, UInt size) const;
+  template <typename... Ns> inline decltype(auto) begin(Ns... n) const;
+  template <typename... Ns> inline decltype(auto) end(Ns... n) const;
+
+  template <typename... Ns>
+  inline decltype(auto) begin_reinterpret(Ns... n);
+  template <typename... Ns>
+  inline decltype(auto) end_reinterpret(Ns... n);
+
+  template <typename... Ns>
+  inline decltype(auto) begin_reinterpret(Ns... n) const;
+  template <typename... Ns>
+  inline decltype(auto) end_reinterpret(Ns... n) const;
 
   /* ------------------------------------------------------------------------ */
   /* Methods                                                                  */
   /* ------------------------------------------------------------------------ */
 public:
   /// append a tuple of size nb_component containing value
   inline void push_back(const_reference value);
   /// append a vector
   // inline void push_back(const value_type new_elem[]);
 
   /// append a Vector or a Matrix
   template <template <typename> class C>
   inline void push_back(const C<T> & new_elem);
   /// append the value of the iterator
   template <typename Ret> inline void push_back(const iterator<Ret> & it);
 
   /// erase the value at position i
   inline void erase(UInt i);
   /// ask Nico, clarify
   template <typename R> inline iterator<R> erase(const iterator<R> & it);
 
   /// changes the allocated size but not the size
   virtual void reserve(UInt size);
 
   /// change the size of the Array
   virtual void resize(UInt size);
 
   /// change the size of the Array and initialize the values
   virtual void resize(UInt size, const T & val);
 
   /// change the number of components by interlacing data
   /// @param multiplicator number of interlaced components add
   /// @param block_size blocks of data in the array
   /// Examaple for block_size = 2, multiplicator = 2
   /// array = oo oo oo -> new array = oo nn nn oo nn nn oo nn nn
   void extendComponentsInterlaced(UInt multiplicator, UInt stride);
 
   /// search elem in the vector, return  the position of the first occurrence or
   /// -1 if not found
   UInt find(const_reference elem) const;
 
   /// @see Array::find(const_reference elem) const
   UInt find(T elem[]) const;
 
   /// @see Array::find(const_reference elem) const
   template <template <typename> class C> inline UInt find(const C<T> & elem);
 
   /// set all entries of the array to 0
   inline void clear() { std::fill_n(values, size_ * nb_component, T()); }
 
   /// set all entries of the array to the value t
   /// @param t value to fill the array with
   inline void set(T t) { std::fill_n(values, size_ * nb_component, t); }
 
   /// set all tuples of the array to a given vector or matrix
   /// @param vm Matrix or Vector to fill the array with
   template <template <typename> class C> inline void set(const C<T> & vm);
 
   /// Append the content of the other array to the current one
   void append(const Array<T> & other);
 
   /// copy another Array in the current Array, the no_sanity_check allows you to
   /// force the copy in cases where you know what you do with two non matching
   /// Arrays in terms of n
   void copy(const Array<T, is_scal> & other, bool no_sanity_check = false);
 
   /// give the address of the memory allocated for this vector
   T * storage() const { return values; };
 
   /// function to print the containt of the class
   void printself(std::ostream & stream, int indent = 0) const override;
 
 protected:
   /// perform the allocation for the constructors
   void allocate(UInt size, UInt nb_component = 1);
 
   /// resize initializing with uninitialized_fill if fill is set
   void resizeUnitialized(UInt new_size, bool fill, const T & val = T());
 
   /* ------------------------------------------------------------------------ */
   /* Operators                                                                */
   /* ------------------------------------------------------------------------ */
 public:
   /// substraction entry-wise
   Array<T, is_scal> & operator-=(const Array<T, is_scal> & other);
   /// addition entry-wise
   Array<T, is_scal> & operator+=(const Array<T, is_scal> & other);
   /// multiply evry entry by alpha
   Array<T, is_scal> & operator*=(const T & alpha);
 
   /// check if the array are identical entry-wise
   bool operator==(const Array<T, is_scal> & other) const;
   /// @see Array::operator==(const Array<T, is_scal> & other) const
   bool operator!=(const Array<T, is_scal> & other) const;
 
   /// return a reference to the j-th entry of the i-th tuple
   inline reference operator()(UInt i, UInt j = 0);
   /// return a const reference to the j-th entry of the i-th tuple
   inline const_reference operator()(UInt i, UInt j = 0) const;
 
   /// return a reference to the ith component of the 1D array
   inline reference operator[](UInt i);
   /// return a const reference to the ith component of the 1D array
   inline const_reference operator[](UInt i) const;
 
   /* ------------------------------------------------------------------------ */
   /* Class Members                                                            */
   /* ------------------------------------------------------------------------ */
 protected:
   /// array of values
   T * values; // /!\ very dangerous
 };
 
 /* -------------------------------------------------------------------------- */
 /* Inline Functions Array<T, is_scal>                                         */
 /* -------------------------------------------------------------------------- */
 template <typename T, bool is_scal>
 inline std::ostream & operator<<(std::ostream & stream,
                                  const Array<T, is_scal> & _this) {
   _this.printself(stream);
   return stream;
 }
 
 /* -------------------------------------------------------------------------- */
 /* Inline Functions ArrayBase                                                 */
 /* -------------------------------------------------------------------------- */
 inline std::ostream & operator<<(std::ostream & stream,
                                  const ArrayBase & _this) {
   _this.printself(stream);
   return stream;
 }
 
 } // namespace akantu
 
 #include "aka_array_tmpl.hh"
 #include "aka_types.hh"
 
 #endif /* __AKANTU_VECTOR_HH__ */
diff --git a/src/common/aka_array_tmpl.hh b/src/common/aka_array_tmpl.hh
index dcbbfffc0..5c3a508ea 100644
--- a/src/common/aka_array_tmpl.hh
+++ b/src/common/aka_array_tmpl.hh
@@ -1,1543 +1,1250 @@
 /**
  * @file   aka_array_tmpl.hh
  *
  * @author Nicolas Richart <nicolas.richart@epfl.ch>
  *
  * @date creation: Thu Jul 15 2010
  * @date last modification: Fri Jan 22 2016
  *
  * @brief  Inline functions of the classes Array<T> and ArrayBase
  *
  * @section LICENSE
  *
  * Copyright (©)  2010-2012, 2014,  2015 EPFL  (Ecole Polytechnique  Fédérale de
  * Lausanne)  Laboratory (LSMS  -  Laboratoire de  Simulation  en Mécanique  des
  * Solides)
  *
  * Akantu is free  software: you can redistribute it and/or  modify it under the
  * terms  of the  GNU Lesser  General Public  License as  published by  the Free
  * Software Foundation, either version 3 of the License, or (at your option) any
  * later version.
  *
  * Akantu is  distributed in the  hope that it  will be useful, but  WITHOUT ANY
  * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
  * A  PARTICULAR PURPOSE. See  the GNU  Lesser General  Public License  for more
  * details.
  *
  * You should  have received  a copy  of the GNU  Lesser General  Public License
  * along with Akantu. If not, see <http://www.gnu.org/licenses/>.
  *
  */
 
 /* -------------------------------------------------------------------------- */
 /* Inline Functions Array<T>                                                 */
 /* -------------------------------------------------------------------------- */
 #include "aka_array.hh"
 /* -------------------------------------------------------------------------- */
 #include <memory>
 /* -------------------------------------------------------------------------- */
 
 #ifndef __AKANTU_AKA_ARRAY_TMPL_HH__
 #define __AKANTU_AKA_ARRAY_TMPL_HH__
 
 namespace akantu {
 
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
 inline T & Array<T, is_scal>::operator()(UInt i, UInt j) {
   AKANTU_DEBUG_ASSERT(size_ > 0, "The array \"" << id << "\" is empty");
   AKANTU_DEBUG_ASSERT((i < size_) && (j < nb_component),
                       "The value at position ["
                           << i << "," << j << "] is out of range in array \""
                           << id << "\"");
   return values[i * nb_component + j];
 }
 
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
 inline const T & Array<T, is_scal>::operator()(UInt i, UInt j) const {
   AKANTU_DEBUG_ASSERT(size_ > 0, "The array \"" << id << "\" is empty");
   AKANTU_DEBUG_ASSERT((i < size_) && (j < nb_component),
                       "The value at position ["
                           << i << "," << j << "] is out of range in array \""
                           << id << "\"");
   return values[i * nb_component + j];
 }
 
 template <class T, bool is_scal>
 inline T & Array<T, is_scal>::operator[](UInt i) {
   AKANTU_DEBUG_ASSERT(size_ > 0, "The array \"" << id << "\" is empty");
   AKANTU_DEBUG_ASSERT((i < size_ * nb_component),
                       "The value at position ["
                           << i << "] is out of range in array \"" << id
                           << "\"");
   return values[i];
 }
 
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
 inline const T & Array<T, is_scal>::operator[](UInt i) const {
   AKANTU_DEBUG_ASSERT(size_ > 0, "The array \"" << id << "\" is empty");
   AKANTU_DEBUG_ASSERT((i < size_ * nb_component),
                       "The value at position ["
                           << i << "] is out of range in array \"" << id
                           << "\"");
   return values[i];
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * append a tuple to the array with the value value for all components
  * @param value the new last tuple or the array will contain nb_component copies
  * of value
  */
 template <class T, bool is_scal>
 inline void Array<T, is_scal>::push_back(const T & value) {
   resizeUnitialized(size_ + 1, true, value);
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * append a tuple to the array
  * @param new_elem a C-array containing the values to be copied to the end of
  * the array */
 // template <class T, bool is_scal>
 // inline void Array<T, is_scal>::push_back(const T new_elem[]) {
 //   UInt pos = size_;
 
 //   resizeUnitialized(size_ + 1, false);
 
 //   T * tmp = values + nb_component * pos;
 //   std::uninitialized_copy(new_elem, new_elem + nb_component, tmp);
 // }
 
 /* -------------------------------------------------------------------------- */
 /**
  * append a matrix or a vector to the array
  * @param new_elem a reference to a Matrix<T> or Vector<T> */
 template <class T, bool is_scal>
 template <template <typename> class C>
 inline void Array<T, is_scal>::push_back(const C<T> & new_elem) {
   AKANTU_DEBUG_ASSERT(
       nb_component == new_elem.size(),
       "The vector("
           << new_elem.size()
           << ") as not a size compatible with the Array (nb_component="
           << nb_component << ").");
   UInt pos = size_;
   resizeUnitialized(size_ + 1, false);
 
   T * tmp = values + nb_component * pos;
   std::uninitialized_copy(new_elem.storage(), new_elem.storage() + nb_component,
                           tmp);
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * append a tuple to the array
  * @param it an iterator to the tuple to be copied to the end of the array */
 template <class T, bool is_scal>
 template <class Ret>
 inline void
 Array<T, is_scal>::push_back(const Array<T, is_scal>::iterator<Ret> & it) {
   UInt pos = size_;
 
   resizeUnitialized(size_ + 1, false);
 
   T * tmp = values + nb_component * pos;
   T * new_elem = it.data();
   std::uninitialized_copy(new_elem, new_elem + nb_component, tmp);
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * erase an element. If the erased element is not the last of the array, the
  * last element is moved into the hole in order to maintain contiguity. This
  * may invalidate existing iterators (For instance an iterator obtained by
  * Array::end() is no longer correct) and will change the order of the
  * elements.
  * @param i index of element to erase
  */
 template <class T, bool is_scal> inline void Array<T, is_scal>::erase(UInt i) {
   AKANTU_DEBUG_IN();
   AKANTU_DEBUG_ASSERT((size_ > 0), "The array is empty");
   AKANTU_DEBUG_ASSERT((i < size_), "The element at position ["
                                        << i << "] is out of range (" << i
                                        << ">=" << size_ << ")");
 
   if (i != (size_ - 1)) {
     for (UInt j = 0; j < nb_component; ++j) {
       values[i * nb_component + j] = values[(size_ - 1) * nb_component + j];
     }
   }
 
   resize(size_ - 1);
   AKANTU_DEBUG_OUT();
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * Subtract another array entry by entry from this array in place. Both arrays
  * must
  * have the same size and nb_component. If the arrays have different shapes,
  * code compiled in debug mode will throw an expeption and optimised code
  * will behave in an unpredicted manner
  * @param other array to subtract from this
  * @return reference to modified this
  */
 template <class T, bool is_scal>
 Array<T, is_scal> & Array<T, is_scal>::
 operator-=(const Array<T, is_scal> & vect) {
   AKANTU_DEBUG_ASSERT((size_ == vect.size_) &&
                           (nb_component == vect.nb_component),
                       "The too array don't have the same sizes");
 
   T * a = values;
   T * b = vect.storage();
   for (UInt i = 0; i < size_ * nb_component; ++i) {
     *a -= *b;
     ++a;
     ++b;
   }
 
   return *this;
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * Add another array entry by entry to this array in place. Both arrays must
  * have the same size and nb_component. If the arrays have different shapes,
  * code compiled in debug mode will throw an expeption and optimised code
  * will behave in an unpredicted manner
  * @param other array to add to this
  * @return reference to modified this
  */
 template <class T, bool is_scal>
 Array<T, is_scal> & Array<T, is_scal>::
 operator+=(const Array<T, is_scal> & vect) {
   AKANTU_DEBUG_ASSERT((size_ == vect.size) &&
                           (nb_component == vect.nb_component),
                       "The too array don't have the same sizes");
 
   T * a = values;
   T * b = vect.storage();
   for (UInt i = 0; i < size_ * nb_component; ++i) {
     *a++ += *b++;
   }
 
   return *this;
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * Multiply all entries of this array by a scalar in place
  * @param alpha scalar multiplicant
  * @return reference to modified this
  */
 template <class T, bool is_scal>
 Array<T, is_scal> & Array<T, is_scal>::operator*=(const T & alpha) {
   T * a = values;
   for (UInt i = 0; i < size_ * nb_component; ++i) {
     *a++ *= alpha;
   }
 
   return *this;
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * Compare this array element by element to another.
  * @param other array to compare to
  * @return true it all element are equal and arrays have the same shape, else
  * false
  */
 template <class T, bool is_scal>
 bool Array<T, is_scal>::operator==(const Array<T, is_scal> & array) const {
   bool equal = nb_component == array.nb_component && size_ == array.size_ &&
                id == array.id;
   if (!equal)
     return false;
 
   if (values == array.storage())
     return true;
   else
     return std::equal(values, values + size_ * nb_component, array.storage());
 }
 
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
 bool Array<T, is_scal>::operator!=(const Array<T, is_scal> & array) const {
   return !operator==(array);
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * set all tuples of the array to a given vector or matrix
  * @param vm Matrix or Vector to fill the array with
  */
 template <class T, bool is_scal>
 template <template <typename> class C>
 inline void Array<T, is_scal>::set(const C<T> & vm) {
   AKANTU_DEBUG_ASSERT(
       nb_component == vm.size(),
       "The size of the object does not match the number of components");
   for (T * it = values; it < values + nb_component * size_;
        it += nb_component) {
     std::copy(vm.storage(), vm.storage() + nb_component, it);
   }
 }
 
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
 void Array<T, is_scal>::append(const Array<T> & other) {
   AKANTU_DEBUG_ASSERT(
       nb_component == other.nb_component,
       "Cannot append an array with a different number of component");
   UInt old_size = this->size_;
   this->resizeUnitialized(this->size_ + other.size(), false);
 
   T * tmp = values + nb_component * old_size;
   std::uninitialized_copy(other.storage(),
                           other.storage() + other.size() * nb_component, tmp);
 }
 
 /* -------------------------------------------------------------------------- */
 /* Functions Array<T, is_scal>                                                */
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
 Array<T, is_scal>::Array(UInt size, UInt nb_component, const ID & id)
     : ArrayBase(id), values(NULL) {
   AKANTU_DEBUG_IN();
   allocate(size, nb_component);
 
   if (!is_scal) {
     T val = T();
     std::uninitialized_fill(values, values + size * nb_component, val);
   }
 
   AKANTU_DEBUG_OUT();
 }
 
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
 Array<T, is_scal>::Array(UInt size, UInt nb_component, const T def_values[],
                          const ID & id)
     : ArrayBase(id), values(NULL) {
   AKANTU_DEBUG_IN();
   allocate(size, nb_component);
 
   T * tmp = values;
 
   for (UInt i = 0; i < size; ++i) {
     tmp = values + nb_component * i;
     std::uninitialized_copy(def_values, def_values + nb_component, tmp);
   }
   AKANTU_DEBUG_OUT();
 }
 
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
 Array<T, is_scal>::Array(UInt size, UInt nb_component, const T & value,
                          const ID & id)
     : ArrayBase(id), values(NULL) {
   AKANTU_DEBUG_IN();
   allocate(size, nb_component);
 
   std::uninitialized_fill_n(values, size * nb_component, value);
 
   AKANTU_DEBUG_OUT();
 }
 
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
 Array<T, is_scal>::Array(const Array<T, is_scal> & vect, bool deep,
                          const ID & id)
     : ArrayBase(vect) {
   AKANTU_DEBUG_IN();
   this->id = (id == "") ? vect.id : id;
 
   if (deep) {
     allocate(vect.size_, vect.nb_component);
     T * tmp = values;
     std::uninitialized_copy(vect.storage(),
                             vect.storage() + size_ * nb_component, tmp);
   } else {
     this->values = vect.storage();
     this->size_ = vect.size_;
     this->nb_component = vect.nb_component;
     this->allocated_size = vect.allocated_size;
     this->size_of_type = vect.size_of_type;
   }
 
   AKANTU_DEBUG_OUT();
 }
 
 /* -------------------------------------------------------------------------- */
 #ifndef SWIG
 template <class T, bool is_scal>
 Array<T, is_scal>::Array(const std::vector<T> & vect) {
   AKANTU_DEBUG_IN();
   this->id = "";
 
   allocate(vect.size(), 1);
   T * tmp = values;
   std::uninitialized_copy(&(vect[0]), &(vect[size_ - 1]), tmp);
 
   AKANTU_DEBUG_OUT();
 }
 #endif
 
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal> Array<T, is_scal>::~Array() {
   AKANTU_DEBUG_IN();
   AKANTU_DEBUG(dblAccessory,
                "Freeing " << printMemorySize<T>(allocated_size * nb_component)
                           << " (" << id << ")");
 
   if (values) {
     if (!is_scal)
       for (UInt i = 0; i < size_ * nb_component; ++i) {
         T * obj = values + i;
         obj->~T();
       }
     free(values);
   }
   size_ = allocated_size = 0;
   AKANTU_DEBUG_OUT();
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * perform the allocation for the constructors
  * @param size is the size of the array
  * @param nb_component is the number of component of the array
  */
 template <class T, bool is_scal>
 void Array<T, is_scal>::allocate(UInt size, UInt nb_component) {
   AKANTU_DEBUG_IN();
   if (size == 0) {
     values = nullptr;
   } else {
     values = static_cast<T *>(malloc(nb_component * size * sizeof(T)));
     AKANTU_DEBUG_ASSERT(values != nullptr,
                         "Cannot allocate "
                             << printMemorySize<T>(size * nb_component) << " ("
                             << id << ")");
   }
 
   if (values == NULL) {
     this->size_ = this->allocated_size = 0;
   } else {
     AKANTU_DEBUG(dblAccessory, "Allocated "
                                    << printMemorySize<T>(size * nb_component)
                                    << " (" << id << ")");
     this->size_ = this->allocated_size = size;
   }
 
   this->size_of_type = sizeof(T);
   this->nb_component = nb_component;
 
   AKANTU_DEBUG_OUT();
 }
 
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
 void Array<T, is_scal>::reserve(UInt new_size) {
   UInt tmp_size = this->size_;
   resizeUnitialized(new_size, false);
   this->size_ = tmp_size;
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * change the size of the array and allocate or free memory if needed. If the
  * size increases, the new tuples are filled with zeros
  * @param new_size new number of tuples contained in the array */
 template <class T, bool is_scal> void Array<T, is_scal>::resize(UInt new_size) {
   resizeUnitialized(new_size, !is_scal);
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * change the size of the array and allocate or free memory if needed. If the
  * size increases, the new tuples are filled with zeros
  * @param new_size new number of tuples contained in the array */
 template <class T, bool is_scal>
 void Array<T, is_scal>::resize(UInt new_size, const T & val) {
   this->resizeUnitialized(new_size, true, val);
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * change the size of the array and allocate or free memory if needed.
  * @param new_size new number of tuples contained in the array */
 template <class T, bool is_scal>
 void Array<T, is_scal>::resizeUnitialized(UInt new_size, bool fill,
                                           const T & val) {
   //  AKANTU_DEBUG_IN();
   // free some memory
   if (new_size <= allocated_size) {
     if (!is_scal) {
       T * old_values = values;
       if (new_size < size_) {
         for (UInt i = new_size * nb_component; i < size_ * nb_component; ++i) {
           T * obj = old_values + i;
           obj->~T();
         }
       }
     }
 
     if (allocated_size - new_size > AKANTU_MIN_ALLOCATION) {
       AKANTU_DEBUG(dblAccessory,
                    "Freeing " << printMemorySize<T>((allocated_size - size_) *
                                                     nb_component)
                               << " (" << id << ")");
 
       // Normally there are no allocation problem when reducing an array
       if (new_size == 0) {
         free(values);
         values = NULL;
       } else {
         auto * tmp_ptr = static_cast<T *>(
             realloc(values, new_size * nb_component * sizeof(T)));
 
         if (tmp_ptr == NULL) {
           AKANTU_EXCEPTION("Cannot free data ("
                            << id << ")"
                            << " [current allocated size : " << allocated_size
                            << " | "
                            << "requested size : " << new_size << "]");
         }
         values = tmp_ptr;
       }
       allocated_size = new_size;
     }
   } else {
     // allocate more memory
     UInt size_to_alloc = (new_size - allocated_size < AKANTU_MIN_ALLOCATION)
                              ? allocated_size + AKANTU_MIN_ALLOCATION
                              : new_size;
 
     auto * tmp_ptr = static_cast<T *>(
         realloc(values, size_to_alloc * nb_component * sizeof(T)));
     AKANTU_DEBUG_ASSERT(
         tmp_ptr != NULL,
         "Cannot allocate " << printMemorySize<T>(size_to_alloc * nb_component));
     if (tmp_ptr == NULL) {
       AKANTU_DEBUG_ERROR("Cannot allocate more data ("
                          << id << ")"
                          << " [current allocated size : " << allocated_size
                          << " | "
                          << "requested size : " << new_size << "]");
     }
 
     AKANTU_DEBUG(dblAccessory,
                  "Allocating " << printMemorySize<T>(
                      (size_to_alloc - allocated_size) * nb_component));
 
     allocated_size = size_to_alloc;
     values = tmp_ptr;
   }
 
   if (fill && this->size_ < new_size) {
     std::uninitialized_fill(values + size_ * nb_component,
                             values + new_size * nb_component, val);
   }
 
   size_ = new_size;
   //  AKANTU_DEBUG_OUT();
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * change the number of components by interlacing data
  * @param multiplicator number of interlaced components add
  * @param block_size blocks of data in the array
  * Examaple for block_size = 2, multiplicator = 2
  * array = oo oo oo -> new array = oo nn nn oo nn nn oo nn nn */
 template <class T, bool is_scal>
 void Array<T, is_scal>::extendComponentsInterlaced(UInt multiplicator,
                                                    UInt block_size) {
   AKANTU_DEBUG_IN();
 
   if (multiplicator == 1)
     return;
 
   AKANTU_DEBUG_ASSERT(multiplicator > 1, "invalid multiplicator");
   AKANTU_DEBUG_ASSERT(nb_component % block_size == 0,
                       "stride must divide actual number of components");
 
   values = static_cast<T *>(
       realloc(values, nb_component * multiplicator * size_ * sizeof(T)));
 
   UInt new_component = nb_component / block_size * multiplicator;
 
   for (UInt i = 0, k = size_ - 1; i < size_; ++i, --k) {
     for (UInt j = 0; j < new_component; ++j) {
       UInt m = new_component - j - 1;
       UInt n = m / multiplicator;
       for (UInt l = 0, p = block_size - 1; l < block_size; ++l, --p) {
         values[k * nb_component * multiplicator + m * block_size + p] =
             values[k * nb_component + n * block_size + p];
       }
     }
   }
 
   nb_component = nb_component * multiplicator;
 
   AKANTU_DEBUG_OUT();
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * search elem in the array, return  the position of the first occurrence or
  * -1 if not found
  *  @param elem the element to look for
  *  @return index of the first occurrence of elem or -1 if elem is not present
  */
 template <class T, bool is_scal>
 UInt Array<T, is_scal>::find(const T & elem) const {
   AKANTU_DEBUG_IN();
 
   auto begin = this->begin();
   auto end = this->end();
   auto it = std::find(begin, end, elem);
 
   AKANTU_DEBUG_OUT();
   return (it != end) ? it - begin : UInt(-1);
 }
 
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal> UInt Array<T, is_scal>::find(T elem[]) const {
   AKANTU_DEBUG_IN();
   T * it = values;
   UInt i = 0;
   for (; i < size_; ++i) {
     if (*it == elem[0]) {
       T * cit = it;
       UInt c = 0;
       for (; (c < nb_component) && (*cit == elem[c]); ++c, ++cit)
         ;
       if (c == nb_component) {
         AKANTU_DEBUG_OUT();
         return i;
       }
     }
     it += nb_component;
   }
   return UInt(-1);
 }
 
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
 template <template <typename> class C>
 inline UInt Array<T, is_scal>::find(const C<T> & elem) {
   AKANTU_DEBUG_ASSERT(elem.size() == nb_component,
                       "Cannot find an element with a wrong size ("
                           << elem.size() << ") != " << nb_component);
   return this->find(elem.storage());
 }
 
 /* -------------------------------------------------------------------------- */
 /**
  * copy the content of another array. This overwrites the current content.
  * @param other Array to copy into this array. It has to have the same
  * nb_component as this. If compiled in debug mode, an incorrect other will
  * result in an exception being thrown. Optimised code may result in
  * unpredicted behaviour.
  */
 template <class T, bool is_scal>
 void Array<T, is_scal>::copy(const Array<T, is_scal> & vect,
                              bool no_sanity_check) {
   AKANTU_DEBUG_IN();
 
   if (!no_sanity_check)
     if (vect.nb_component != nb_component)
       AKANTU_DEBUG_ERROR(
           "The two arrays do not have the same number of components");
 
   resize((vect.size_ * vect.nb_component) / nb_component);
 
   T * tmp = values;
   std::uninitialized_copy(vect.storage(), vect.storage() + size_ * nb_component,
                           tmp);
 
   AKANTU_DEBUG_OUT();
 }
 
 /* -------------------------------------------------------------------------- */
 template <bool is_scal> class ArrayPrintHelper {
 public:
   template <typename T>
   static void print_content(const Array<T> & vect, std::ostream & stream,
                             int indent) {
     if (AKANTU_DEBUG_TEST(dblDump) || AKANTU_DEBUG_LEVEL_IS_TEST()) {
       std::string space;
       for (Int i = 0; i < indent; i++, space += AKANTU_INDENT)
         ;
 
       stream << space << " + values         : {";
       for (UInt i = 0; i < vect.size(); ++i) {
         stream << "{";
         for (UInt j = 0; j < vect.getNbComponent(); ++j) {
           stream << vect(i, j);
           if (j != vect.getNbComponent() - 1)
             stream << ", ";
         }
         stream << "}";
         if (i != vect.size() - 1)
           stream << ", ";
       }
       stream << "}" << std::endl;
     }
   }
 };
 
 template <> class ArrayPrintHelper<false> {
 public:
   template <typename T>
   static void print_content(__attribute__((unused)) const Array<T> & vect,
                             __attribute__((unused)) std::ostream & stream,
                             __attribute__((unused)) int indent) {}
 };
 
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
 void Array<T, is_scal>::printself(std::ostream & stream, int indent) const {
   std::string space;
   for (Int i = 0; i < indent; i++, space += AKANTU_INDENT)
     ;
 
   std::streamsize prec = stream.precision();
   std::ios_base::fmtflags ff = stream.flags();
 
   stream.setf(std::ios_base::showbase);
   stream.precision(2);
 
   stream << space << "Array<" << debug::demangle(typeid(T).name()) << "> ["
          << std::endl;
   stream << space << " + id             : " << this->id << std::endl;
   stream << space << " + size           : " << this->size_ << std::endl;
   stream << space << " + nb_component   : " << this->nb_component << std::endl;
   stream << space << " + allocated size : " << this->allocated_size
          << std::endl;
   stream << space << " + memory size    : "
          << printMemorySize<T>(allocated_size * nb_component) << std::endl;
   if (!AKANTU_DEBUG_LEVEL_IS_TEST())
     stream << space << " + address        : " << std::hex << this->values
            << std::dec << std::endl;
 
   stream.precision(prec);
   stream.flags(ff);
 
   ArrayPrintHelper<is_scal>::print_content(*this, stream, indent);
 
   stream << space << "]" << std::endl;
 }
 
 /* -------------------------------------------------------------------------- */
 /* Inline Functions ArrayBase                                                */
 /* -------------------------------------------------------------------------- */
 
 inline UInt ArrayBase::getMemorySize() const {
   return allocated_size * nb_component * size_of_type;
 }
 
 inline void ArrayBase::empty() { size_ = 0; }
 
 /* -------------------------------------------------------------------------- */
 /* Iterators                                                                  */
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
 template <class R, class IR, bool is_r_scal>
 class Array<T, is_scal>::iterator_internal {
 public:
   using value_type = R;
   using pointer = R *;
   using reference = R &;
   using proxy = typename R::proxy;
   using const_proxy = const typename R::proxy;
   using const_reference = const R &;
   using internal_value_type = IR;
   using internal_pointer = IR *;
   using difference_type = std::ptrdiff_t;
   using iterator_category = std::random_access_iterator_tag;
 
 public:
   iterator_internal() : initial(NULL), ret(NULL), ret_ptr(NULL){};
 
   iterator_internal(pointer_type data, UInt _offset)
       : _offset(_offset), initial(data), ret(NULL), ret_ptr(data) {
     AKANTU_DEBUG_ERROR(
         "The constructor should never be called it is just an ugly trick...");
   }
 
   iterator_internal(pointer wrapped)
       : _offset(wrapped->size()), initial(wrapped->storage()),
         ret(const_cast<internal_pointer>(wrapped)),
         ret_ptr(wrapped->storage()) {}
 
   iterator_internal(const iterator_internal & it) {
     if (this != &it) {
       this->_offset = it._offset;
       this->initial = it.initial;
       this->ret_ptr = it.ret_ptr;
       this->ret = new internal_value_type(*it.ret, false);
     }
   }
 
   virtual ~iterator_internal() { delete ret; };
 
   inline iterator_internal & operator=(const iterator_internal & it) {
     if (this != &it) {
       this->_offset = it._offset;
       this->initial = it.initial;
       this->ret_ptr = it.ret_ptr;
       if (this->ret)
         this->ret->shallowCopy(*it.ret);
       else
         this->ret = new internal_value_type(*it.ret, false);
     }
     return *this;
   }
 
   UInt getCurrentIndex() {
     return (this->ret_ptr - this->initial) / this->_offset;
   };
 
   inline reference operator*() {
     ret->values = ret_ptr;
     return *ret;
   };
   inline const_reference operator*() const {
     ret->values = ret_ptr;
     return *ret;
   };
   inline pointer operator->() {
     ret->values = ret_ptr;
     return ret;
   };
   inline iterator_internal & operator++() {
     ret_ptr += _offset;
     return *this;
   };
   inline iterator_internal & operator--() {
     ret_ptr -= _offset;
     return *this;
   };
 
   inline iterator_internal & operator+=(const UInt n) {
     ret_ptr += _offset * n;
     return *this;
   }
   inline iterator_internal & operator-=(const UInt n) {
     ret_ptr -= _offset * n;
     return *this;
   }
 
   inline proxy operator[](const UInt n) {
     ret->values = ret_ptr + n * _offset;
     return proxy(*ret);
   }
   inline const_proxy operator[](const UInt n) const {
     ret->values = ret_ptr + n * _offset;
     return const_proxy(*ret);
   }
 
   inline bool operator==(const iterator_internal & other) const {
     return this->ret_ptr == other.ret_ptr;
   }
   inline bool operator!=(const iterator_internal & other) const {
     return this->ret_ptr != other.ret_ptr;
   }
   inline bool operator<(const iterator_internal & other) const {
     return this->ret_ptr < other.ret_ptr;
   }
   inline bool operator<=(const iterator_internal & other) const {
     return this->ret_ptr <= other.ret_ptr;
   }
   inline bool operator>(const iterator_internal & other) const {
     return this->ret_ptr > other.ret_ptr;
   }
   inline bool operator>=(const iterator_internal & other) const {
     return this->ret_ptr >= other.ret_ptr;
   }
 
   inline iterator_internal operator+(difference_type n) {
     iterator_internal tmp(*this);
     tmp += n;
     return tmp;
   }
   inline iterator_internal operator-(difference_type n) {
     iterator_internal tmp(*this);
     tmp -= n;
     return tmp;
   }
 
   inline difference_type operator-(const iterator_internal & b) {
     return (this->ret_ptr - b.ret_ptr) / _offset;
   }
 
   inline pointer_type data() const { return ret_ptr; }
   inline difference_type offset() const { return _offset; }
 
 protected:
   UInt _offset{0};
   pointer_type initial;
   internal_pointer ret;
   pointer_type ret_ptr;
 };
 
 /* -------------------------------------------------------------------------- */
 /**
  * Specialization for scalar types
  */
 template <class T, bool is_scal>
 template <class R, class IR>
 class Array<T, is_scal>::iterator_internal<R, IR, true> {
 public:
   using value_type = R;
   using pointer = R *;
   using reference = R &;
   using const_reference = const R &;
   using internal_value_type = IR;
   using internal_pointer = IR *;
   using difference_type = std::ptrdiff_t;
   using iterator_category = std::random_access_iterator_tag;
 
 public:
   iterator_internal(pointer data = nullptr, UInt _offset = 1)
       : _offset(_offset), ret(data), initial(data){};
   iterator_internal(const iterator_internal & it) = default;
   iterator_internal(iterator_internal && it) = default;
 
   virtual ~iterator_internal() = default;
 
   inline iterator_internal & operator=(const iterator_internal & it) = default;
 
   UInt getCurrentIndex() {
     return (this->ret - this->initial) / this->_offset;
   };
 
   inline reference operator*() { return *ret; };
   inline const_reference operator*() const { return *ret; };
   inline pointer operator->() { return ret; };
   inline iterator_internal & operator++() {
     ++ret;
     return *this;
   };
   inline iterator_internal & operator--() {
     --ret;
     return *this;
   };
 
   inline iterator_internal & operator+=(const UInt n) {
     ret += n;
     return *this;
   }
   inline iterator_internal & operator-=(const UInt n) {
     ret -= n;
     return *this;
   }
 
   inline reference operator[](const UInt n) { return ret[n]; }
 
   inline bool operator==(const iterator_internal & other) const {
     return ret == other.ret;
   }
   inline bool operator!=(const iterator_internal & other) const {
     return ret != other.ret;
   }
   inline bool operator<(const iterator_internal & other) const {
     return ret < other.ret;
   }
   inline bool operator<=(const iterator_internal & other) const {
     return ret <= other.ret;
   }
   inline bool operator>(const iterator_internal & other) const {
     return ret > other.ret;
   }
   inline bool operator>=(const iterator_internal & other) const {
     return ret >= other.ret;
   }
 
   inline iterator_internal operator-(difference_type n) {
     return iterator_internal(ret - n);
   }
   inline iterator_internal operator+(difference_type n) {
     return iterator_internal(ret + n);
   }
 
   inline difference_type operator-(const iterator_internal & b) {
     return ret - b.ret;
   }
 
   inline pointer data() const { return ret; }
   inline difference_type offset() const { return _offset; }
 
 protected:
   difference_type _offset;
   pointer ret;
   pointer initial;
 };
 
 /* -------------------------------------------------------------------------- */
 /* Begin/End functions implementation                                         */
 /* -------------------------------------------------------------------------- */
-/* -------------------------------------------------------------------------- */
-/**
- * Get an iterator that behaves like a pointer akantu::Vector<T> * to the
- * first tuple of the array.
- * @param n vector size. Has to be equal to nb_component. This unfortunate
- * redundancy is necessary to distinguish it from ::begin() which it
- * overloads. If compiled in debug mode, an incorrect value of n will result
- * in an exception being thrown. Optimized code will fail in an unpredicted
- * manner.
- * @return a vector_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::vector_iterator
-Array<T, is_scal>::begin(UInt n) {
-  AKANTU_DEBUG_ASSERT(nb_component == n,
-                      "The iterator is not compatible with the type Array("
-                          << n << ")");
-  return vector_iterator(new Vector<T>(values, n));
-}
-
-/* -------------------------------------------------------------------------- */
-/**
- * Get an iterator that behaves like a pointer akantu::Vector<T> * pointing
- * *past* the last tuple of the array.
- * @param n vector size. see Array::begin(UInt n) for more
- * @return a vector_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::vector_iterator
-Array<T, is_scal>::end(UInt n) {
-  AKANTU_DEBUG_ASSERT(nb_component == n,
-                      "The iterator is not compatible with the type Array("
-                          << n << ")");
-  return vector_iterator(new Vector<T>(values + nb_component * size_, n));
-}
-
-/* -------------------------------------------------------------------------- */
-/**
- * Get a const iterator that behaves like a pointer akantu::Vector<T> * to the
- * first tuple of the array.
- * @param n vector size. see Array::begin(UInt n) for more
- * @return a vector_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::const_vector_iterator
-Array<T, is_scal>::begin(UInt n) const {
-  AKANTU_DEBUG_ASSERT(nb_component == n,
-                      "The iterator is not compatible with the type Array("
-                          << n << ")");
-  return const_vector_iterator(new Vector<T>(values, n));
-}
-
-/* -------------------------------------------------------------------------- */
-/**
- * Get a const iterator that behaves like a pointer akantu::Vector<T> * pointing
- * *past* the last tuple of the array.
- * @param n vector size. see Array::begin(UInt n) for more
- * @return a const_vector_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::const_vector_iterator
-Array<T, is_scal>::end(UInt n) const {
-  AKANTU_DEBUG_ASSERT(nb_component == n,
-                      "The iterator is not compatible with the type Array("
-                          << n << ")");
-  return const_vector_iterator(new Vector<T>(values + nb_component * size_, n));
-}
-
-/* -------------------------------------------------------------------------- */
-/**
- * Get an iterator that behaves like a pointer akantu::Vector<T> * to the
- * first tuple of the array.
- *
- * The reinterpret iterators allow to iterate over an array in any way that
- * preserves the number of entries of the array. This can for instance be use
- * full if the shape of the data in an array is not initially known.
- * @param n vector size.
- * @param size number of tuples in array. n times size must match the number
- * of entries of the array. If compiled in debug mode, an incorrect
- * combination of n and size will result
- * in an exception being thrown. Optimized code will fail in an unpredicted
- * manner.
- * @return a vector_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::vector_iterator
-Array<T, is_scal>::begin_reinterpret(UInt n,
-                                     __attribute__((unused)) UInt size) {
-  AKANTU_DEBUG_ASSERT(n * size == this->nb_component * this->size_,
-                      "The new values for size ("
-                          << size << ") and nb_component (" << n
-                          << ") are not compatible with the one of this array("
-                          << this->size_ << "," << this->nb_component << ")");
-  return vector_iterator(new Vector<T>(values, n));
-}
-
-/* -------------------------------------------------------------------------- */
-/**
- * Get an iterator that behaves like a pointer akantu::Vector<T> * pointing
- * *past* the last tuple of the array.
- * @param n vector size.
- * @param size number of tuples in array. See Array::begin_reinterpret(UInt n,
- * UInt size)
- * @return a vector_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::vector_iterator
-Array<T, is_scal>::end_reinterpret(UInt n, UInt size) {
-  AKANTU_DEBUG_ASSERT(n * size == this->nb_component * this->size_,
-                      "The new values for size ("
-                          << size << ") and nb_component (" << n
-                          << ") are not compatible with the one of this array("
-                          << this->size_ << "," << this->nb_component << ")");
-  return vector_iterator(new Vector<T>(values + n * size, n));
-}
-
-/* -------------------------------------------------------------------------- */
-/**
- * Get a const iterator that behaves like a pointer akantu::Vector<T> * to the
- * first tuple of the array.
- * @param n vector size.
- * @param size number of tuples in array. See Array::begin_reinterpret(UInt n,
- * UInt size)
- * @return a const_vector_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::const_vector_iterator
-Array<T, is_scal>::begin_reinterpret(UInt n,
-                                     __attribute__((unused)) UInt size) const {
-  AKANTU_DEBUG_ASSERT(n * size == this->nb_component * this->size_,
-                      "The new values for size ("
-                          << size << ") and nb_component (" << n
-                          << ") are not compatible with the one of this array("
-                          << this->size_ << "," << this->nb_component << ")");
-  return const_vector_iterator(new Vector<T>(values, n));
-}
-
-/* -------------------------------------------------------------------------- */
-/**
- * Get a const iterator that behaves like a pointer akantu::Vector<T> * pointing
- * *past* the last tuple of the array.
- * @param n vector size.
- * @param size number of tuples in array. See Array::begin_reinterpret(UInt n,
- * UInt size)
- * @return a const_vector_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::const_vector_iterator
-Array<T, is_scal>::end_reinterpret(UInt n, UInt size) const {
-  AKANTU_DEBUG_ASSERT(n * size == this->nb_component * this->size_,
-                      "The new values for size ("
-                          << size << ") and nb_component (" << n
-                          << ") are not compatible with the one of this array("
-                          << this->size_ << "," << this->nb_component << ")");
-  return const_vector_iterator(new Vector<T>(values + n * size, n));
-}
-
-/* -------------------------------------------------------------------------- */
-/**
- * Get an iterator that behaves like a pointer akantu::Matrix<T> * to the
- * first tuple of the array.
- * @param m number of rows
- * @param n number of columns. m times n has to equal nb_component.
- * If compiled in debug mode, an incorrect combination of m and n will result
- * in an exception being thrown. Optimized code will fail in an unpredicted
- * manner.
- * @return a matrix_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::matrix_iterator
-Array<T, is_scal>::begin(UInt m, UInt n) {
-  AKANTU_DEBUG_ASSERT(nb_component == n * m,
-                      "The iterator is not compatible with the type Matrix("
-                          << m << "," << n << ")");
-  return matrix_iterator(new Matrix<T>(values, m, n));
-}
-
-/* -------------------------------------------------------------------------- */
-/**
- * Get an iterator that behaves like a pointer akantu::Matrix<T> * pointing
- * *past* the last tuple of the array.
- * @param m number of rows
- * @param n number of columns. See Array::begin(UInt m, UInt n)
- * @return a matrix_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::matrix_iterator
-Array<T, is_scal>::end(UInt m, UInt n) {
-  AKANTU_DEBUG_ASSERT(nb_component == n * m,
-                      "The iterator is not compatible with the type Matrix("
-                          << m << "," << n << ")");
-  return matrix_iterator(new Matrix<T>(values + nb_component * size_, m, n));
-}
+namespace {
+  template <std::size_t N> struct extract_last {
+    template <typename F, typename... T, typename... Arg>
+    static decltype(auto) extract(F && func, std::tuple<T...> && t,
+                                  Arg... args) {
+      return extract_last<N - 1>::extract(
+          std::forward<F>(func), std::forward<decltype(t)>(t),
+          std::get<sizeof...(T) - N>(std::forward<decltype(t)>(t)), args...);
+    }
+  };
 
-/* -------------------------------------------------------------------------- */
-/**
- * Get a const iterator that behaves like a pointer akantu::Matrix<T> * to the
- * first tuple of the array.
- * @param m number of rows
- * @param n number of columns. See Array::begin(UInt m, UInt n)
- * @return a matrix_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::const_matrix_iterator
-Array<T, is_scal>::begin(UInt m, UInt n) const {
-  AKANTU_DEBUG_ASSERT(nb_component == n * m,
-                      "The iterator is not compatible with the type Matrix("
-                          << m << "," << n << ")");
-  return const_matrix_iterator(new Matrix<T>(values, m, n));
-}
+  template <> struct extract_last<1> {
+    template <typename F, typename... T, typename... Arg>
+    static decltype(auto) extract(F && func, std::tuple<T...> && /*unused*/,
+                                  Arg... args) {
+      return std::forward<F>(func)(args...);
+    }
+  };
 
-/* -------------------------------------------------------------------------- */
-/**
- * Get a const iterator that behaves like a pointer akantu::Matrix<T> * pointing
- * *past* the last tuple of the array.
- * @param m number of rows
- * @param n number of columns. See Array::begin(UInt m, UInt n)
- * @return a const_matrix_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::const_matrix_iterator
-Array<T, is_scal>::end(UInt m, UInt n) const {
-  AKANTU_DEBUG_ASSERT(nb_component == n * m,
-                      "The iterator is not compatible with the type Matrix("
-                          << m << "," << n << ")");
-  return const_matrix_iterator(
-      new Matrix<T>(values + nb_component * size_, m, n));
-}
+#pragma GCC diagnostic push
+#pragma GCC diagnostic ignored "-Wnarrowing"
+  template <std::size_t N> struct InstantiationHelper {
+    template <typename... Ns> static constexpr std::size_t product(Ns... ns) {
+      std::size_t p = 1;
+      for (auto n : std::array<std::size_t, sizeof...(Ns)>{ns...})
+        p *= n;
+      return p;
+    }
 
-/* -------------------------------------------------------------------------- */
-/**
- * Get an iterator that behaves like a pointer akantu::Matrix<T> * to the
- * first tuple of the array.
- *
- * The reinterpret iterators allow to iterate over an array in any way that
- * preserves the number of entries of the array. This can for instance be use
- * full if the shape of the data in an array is not initially known.
- * @param m number of rows
- * @param n number of columns
- * @param size number of tuples in array. m times n times size must match the
- *number
- * of entries of the array. If compiled in debug mode, an incorrect
- * combination of m, n and size will result
- * in an exception being thrown. Optimized code will fail in an unpredicted
- * manner.
- * @return a matrix_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::matrix_iterator
-Array<T, is_scal>::begin_reinterpret(UInt m, UInt n,
-                                     __attribute__((unused)) UInt size) {
-  AKANTU_DEBUG_ASSERT(n * m * size == this->nb_component * this->size_,
-                      "The new values for size ("
-                          << size << ") and nb_component (" << m << "," << n
-                          << " = " << n * m
-                          << ") are not compatible with the one of this array("
-                          << this->size_ << "," << this->nb_component << ")");
-  return matrix_iterator(new Matrix<T>(values, m, n));
-}
+    template <typename... Ns> static std::string to_string(Ns... ns) {
+      std::stringstream sstr;
+      bool first = true;
+      sstr << "(";
+      for (auto n : std::array<std::size_t, sizeof...(Ns)>{ns...}) {
+        if (!first) {
+          sstr << ", ";
+        }
+        sstr << n;
+        first = false;
+      }
+      sstr << ")";
+      return sstr.str();
+    }
+#pragma GCC diagnostic pop
+    template <typename type, typename T, typename... Ns>
+    static auto instantiate(T && data, Ns... ns) {
+      return new type(data, ns...);
+    }
+  };
 
-/* -------------------------------------------------------------------------- */
-/**
- * Get an iterator that behaves like a pointer akantu::Matrix<T> * pointing
- * *past* the last tuple of the array.
- * @param m number of rows
- * @param n number of columns
- * @param size number of tuples in array. See Array::begin_reinterpret(UInt m,
- * UInt n, UInt size)
- * @return a matrix_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::matrix_iterator
-Array<T, is_scal>::end_reinterpret(UInt m, UInt n, UInt size) {
-  AKANTU_DEBUG_ASSERT(n * m * size == this->nb_component * this->size_,
-                      "The new values for size ("
-                          << size << ") and nb_component (" << m << "," << n
-                          << " = " << n * m
-                          << ") are not compatible with the one of this array("
-                          << this->size_ << "," << this->nb_component << ")");
-  return matrix_iterator(new Matrix<T>(values + n * m * size, m, n));
-}
+  template <> struct InstantiationHelper<0> {
+    template <typename type, typename T> static auto instantiate(T && data) {
+      return data;
+    }
 
-/* -------------------------------------------------------------------------- */
-/**
- * Get a const iterator that behaves like a pointer akantu::Matrix<T> * to the
- * first tuple of the array.
- * @param m number of rows
- * @param n number of columns
- * @param size number of tuples in array. See Array::begin_reinterpret(UInt m,
- * UInt n, UInt size)
- * @return a const_matrix_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::const_matrix_iterator
-Array<T, is_scal>::begin_reinterpret(UInt m, UInt n,
-                                     __attribute__((unused)) UInt size) const {
-  AKANTU_DEBUG_ASSERT(n * m * size == this->nb_component * this->size_,
-                      "The new values for size ("
-                          << size << ") and nb_component (" << m << "," << n
-                          << " = " << n * m
-                          << ") are not compatible with the one of this array("
-                          << this->size_ << "," << this->nb_component << ")");
-  return const_matrix_iterator(new Matrix<T>(values, m, n));
-}
+    static constexpr std::size_t product() { return 1; }
+    static std::string to_string() { return ""; }
+  };
 
-/* -------------------------------------------------------------------------- */
-/**
- * Get a const iterator that behaves like a pointer akantu::Matrix<T> * pointing
- * *past* the last tuple of the array.
- * @param m number of rows
- * @param n number of columns
- * @param size number of tuples in array. See Array::begin_reinterpret(UInt m,
- * UInt n, UInt size)
- * @return a const_matrix_iterator
- */
-template <class T, bool is_scal>
-inline typename Array<T, is_scal>::const_matrix_iterator
-Array<T, is_scal>::end_reinterpret(UInt m, UInt n, UInt size) const {
-  AKANTU_DEBUG_ASSERT(n * m * size == this->nb_component * this->size_,
-                      "The new values for size ("
-                          << size << ") and nb_component (" << m << "," << n
-                          << " = " << n * m
-                          << ") are not compatible with the one of this array("
-                          << this->size_ << "," << this->nb_component << ")");
-  return const_matrix_iterator(new Matrix<T>(values + n * m * size, m, n));
-}
+  template <typename Arr, typename T, typename... Ns>
+  decltype(auto) get_iterator(Arr && array, T * data, Ns... ns) {
+    using type = IteratorHelper_t<sizeof...(Ns) -1, T>;
+    using array_type = std::decay_t<Arr>;
+    using iterator =
+        std::conditional_t<std::is_const<Arr>::value,
+                           typename array_type::template const_iterator<type>,
+                           typename array_type::template iterator<type>>;
+    AKANTU_DEBUG_ASSERT(
+        array.getNbComponent() * array.size() ==
+            InstantiationHelper<sizeof...(Ns)>::product(ns...),
+        "The iterator is not compatible with the type "
+            << debug::demangle(typeid(type).name())
+            << InstantiationHelper<sizeof...(Ns)>::to_string(ns...));
+    auto && wrapped = extract_last<sizeof...(Ns)>::extract(
+        [&](auto... n) {
+          return InstantiationHelper<sizeof...(n)>::template instantiate<type>(
+              data, n...);
+        },
+        std::make_tuple(ns...));
+
+    return iterator{wrapped};
+  }
+} // namespace
 
 /* -------------------------------------------------------------------------- */
-/** Get an iterator that behaves like a pointer T * to the
- *  first entry in the member array values
- *  @return a scalar_iterator
- */
 template <class T, bool is_scal>
-inline typename Array<T, is_scal>::scalar_iterator Array<T, is_scal>::begin() {
-  AKANTU_DEBUG_ASSERT(
-      nb_component == 1,
-      "this iterator cannot be used on a vector which has nb_component != 1");
-  return scalar_iterator(values);
+template <typename... Ns>
+inline decltype(auto) Array<T, is_scal>::begin(Ns... ns) {
+  return get_iterator(*this, values, ns..., size_);
 }
 
-/* -------------------------------------------------------------------------- */
-/*! Get an iterator that behaves like a pointer T * that points *past* the
- *  last entry in the member array values
- *  @return a scalar_iterator
- */
 template <class T, bool is_scal>
-inline typename Array<T, is_scal>::scalar_iterator Array<T, is_scal>::end() {
-  AKANTU_DEBUG_ASSERT(
-      nb_component == 1,
-      "this iterator cannot be used on a array which has nb_component != 1");
-  return scalar_iterator(values + size_);
+template <typename... Ns>
+inline decltype(auto) Array<T, is_scal>::end(Ns... ns) {
+  return get_iterator(*this, values + nb_component * size_, ns..., size_);
 }
 
-/* -------------------------------------------------------------------------- */
-/*! Get a const iterator that behaves like a pointer T * to the
- *  first entry in the member array values
- *  @return a const_scalar_iterator
- */
 template <class T, bool is_scal>
-inline typename Array<T, is_scal>::const_scalar_iterator
-Array<T, is_scal>::begin() const {
-  AKANTU_DEBUG_ASSERT(
-      nb_component == 1,
-      "this iterator cannot be used on a array which has nb_component != 1");
-  return const_scalar_iterator(values);
+template <typename... Ns>
+inline decltype(auto) Array<T, is_scal>::begin(Ns... ns) const {
+  return get_iterator(*this, values, ns..., size_);
 }
 
-/* -------------------------------------------------------------------------- */
-/*! Get a const iterator that behaves like a pointer T * that points *past* the
- *  last entry in the member array values
- *  @return a const_scalar_iterator
- */
 template <class T, bool is_scal>
-inline typename Array<T, is_scal>::const_scalar_iterator
-Array<T, is_scal>::end() const {
-  AKANTU_DEBUG_ASSERT(
-      nb_component == 1,
-      "this iterator cannot be used on a array which has nb_component != 1");
-  return const_scalar_iterator(values + size_);
+template <typename... Ns>
+inline decltype(auto) Array<T, is_scal>::end(Ns... ns) const {
+  return get_iterator(*this, values + nb_component * size_, ns..., size_);
 }
 
-/* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
-inline typename Array<T, is_scal>::scalar_iterator
-Array<T, is_scal>::begin_reinterpret(__attribute__((unused)) UInt new_size) {
-  AKANTU_DEBUG_ASSERT(new_size == this->nb_component * this->size_,
-                      "The new values for size ("
-                          << new_size
-                          << ") is not compatible with the one of this array("
-                          << this->size_ << "," << this->nb_component << ")");
-  return scalar_iterator(values);
+template <typename... Ns>
+inline decltype(auto) Array<T, is_scal>::begin_reinterpret(Ns... ns) {
+  return get_iterator(*this, values, ns...);
 }
 
-/* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
-inline typename Array<T, is_scal>::scalar_iterator
-Array<T, is_scal>::end_reinterpret(UInt new_size) {
-  AKANTU_DEBUG_ASSERT(new_size == this->nb_component * this->size_,
-                      "The new values for size ("
-                          << new_size
-                          << ") is not compatible with the one of this array("
-                          << this->size_ << "," << this->nb_component << ")");
-  return scalar_iterator(values + new_size);
+template <typename... Ns>
+inline decltype(auto) Array<T, is_scal>::end_reinterpret(Ns... ns) {
+  return get_iterator(
+      *this, values + InstantiationHelper<sizeof...(Ns)>::product(ns...),
+      ns...);
 }
 
-/* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
-inline typename Array<T, is_scal>::const_scalar_iterator
-Array<T, is_scal>::begin_reinterpret(__attribute__((unused))
-                                     UInt new_size) const {
-  AKANTU_DEBUG_ASSERT(new_size == this->nb_component * this->size_,
-                      "The new values for size ("
-                          << new_size
-                          << ") is not compatible with the one of this array("
-                          << this->size_ << "," << this->nb_component << ")");
-  return const_scalar_iterator(values);
+template <typename... Ns>
+inline decltype(auto) Array<T, is_scal>::begin_reinterpret(Ns... ns) const {
+  return get_iterator(*this, values, ns...);
 }
 
-/* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
-inline typename Array<T, is_scal>::const_scalar_iterator
-Array<T, is_scal>::end_reinterpret(UInt new_size) const {
-  AKANTU_DEBUG_ASSERT(new_size == this->nb_component * this->size_,
-                      "The new values for size ("
-                          << new_size
-                          << ") is not compatible with the one of this array("
-                          << this->size_ << "," << this->nb_component << ")");
-  return const_scalar_iterator(values + new_size);
+template <typename... Ns>
+inline decltype(auto) Array<T, is_scal>::end_reinterpret(Ns... ns) const {
+  return get_iterator(
+      *this, values + InstantiationHelper<sizeof...(Ns)>::product(ns...),
+      ns...);
 }
 
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
 template <typename R>
 class Array<T, is_scal>::const_iterator : public iterator_internal<const R, R> {
 public:
   typedef iterator_internal<const R, R> parent;
   using value_type = typename parent::value_type;
   using pointer = typename parent::pointer;
   using reference = typename parent::reference;
   using difference_type = typename parent::difference_type;
   using iterator_category = typename parent::iterator_category;
 
 public:
   const_iterator() : parent(){};
   const_iterator(pointer_type data, UInt offset) : parent(data, offset) {}
   const_iterator(pointer warped) : parent(warped) {}
   const_iterator(const parent & it) : parent(it) {}
   //    const_iterator(const const_iterator<R> & it) : parent(it) {}
 
   inline const_iterator operator+(difference_type n) {
     return parent::operator+(n);
   }
   inline const_iterator operator-(difference_type n) {
     return parent::operator-(n);
   }
   inline difference_type operator-(const const_iterator & b) {
     return parent::operator-(b);
   }
 
   inline const_iterator & operator++() {
     parent::operator++();
     return *this;
   };
   inline const_iterator & operator--() {
     parent::operator--();
     return *this;
   };
   inline const_iterator & operator+=(const UInt n) {
     parent::operator+=(n);
     return *this;
   }
 };
 // #endif
 
 // #if defined(AKANTU_CORE_CXX11)
 //   template<class R> using iterator = iterator_internal<R>;
 // #else
 template <class T, class R, bool issame = is_same<T, R>::value>
 struct ConstConverterIteratorHelper {
   using const_iterator = typename Array<T>::template const_iterator<R>;
   using iterator = typename Array<T>::template iterator<R>;
   static inline const_iterator convert(const iterator & it) {
     return const_iterator(new R(*it, false));
   }
 };
 
 template <class T, class R> struct ConstConverterIteratorHelper<T, R, true> {
   using const_iterator = typename Array<T>::template const_iterator<R>;
   using iterator = typename Array<T>::template iterator<R>;
   static inline const_iterator convert(const iterator & it) {
     return const_iterator(it.data(), it.offset());
   }
 };
 
 template <class T, bool is_scal>
 template <typename R>
 class Array<T, is_scal>::iterator : public iterator_internal<R> {
 public:
   using parent = iterator_internal<R>;
   using value_type = typename parent::value_type;
   using pointer = typename parent::pointer;
   using reference = typename parent::reference;
   using difference_type = typename parent::difference_type;
   using iterator_category = typename parent::iterator_category;
 
 public:
   iterator() : parent(){};
   iterator(pointer_type data, UInt offset) : parent(data, offset){};
   iterator(pointer warped) : parent(warped) {}
   iterator(const parent & it) : parent(it) {}
   //    iterator(const iterator<R> & it) : parent(it) {}
 
   operator const_iterator<R>() {
     return ConstConverterIteratorHelper<T, R>::convert(*this);
   }
 
   inline iterator operator+(difference_type n) {
     return parent::operator+(n);
     ;
   }
   inline iterator operator-(difference_type n) {
     return parent::operator-(n);
     ;
   }
   inline difference_type operator-(const iterator & b) {
     return parent::operator-(b);
   }
 
   inline iterator & operator++() {
     parent::operator++();
     return *this;
   };
   inline iterator & operator--() {
     parent::operator--();
     return *this;
   };
   inline iterator & operator+=(const UInt n) {
     parent::operator+=(n);
     return *this;
   }
 };
 
 /* -------------------------------------------------------------------------- */
 template <class T, bool is_scal>
 template <typename R>
 inline typename Array<T, is_scal>::template iterator<R>
 Array<T, is_scal>::erase(const iterator<R> & it) {
   T * curr = it.data();
   UInt pos = (curr - values) / nb_component;
   erase(pos);
   iterator<R> rit = it;
   return --rit;
 }
 
 } // namespace akantu
 
 #endif /* __AKANTU_AKA_ARRAY_TMPL_HH__ */