/*============================================================================= Copyright (c) 2011-2016 Bolero MURAKAMI https://github.com/bolero-MURAKAMI/Sprout Distributed under the Boost Software License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) =============================================================================*/ #ifndef SPROUT_ARRAY_ARRAY_HPP #define SPROUT_ARRAY_ARRAY_HPP #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #if SPROUT_USE_INDEX_ITERATOR_IMPLEMENTATION # include #endif namespace sprout { // // array // template class array { public: typedef T value_type; #if SPROUT_USE_INDEX_ITERATOR_IMPLEMENTATION typedef sprout::index_iterator iterator; typedef sprout::index_iterator const_iterator; #else typedef T* iterator; typedef T const* const_iterator; #endif typedef T& reference; typedef T const& const_reference; typedef std::size_t size_type; typedef std::ptrdiff_t difference_type; typedef T* pointer; typedef T const* const_pointer; typedef sprout::reverse_iterator reverse_iterator; typedef sprout::reverse_iterator const_reverse_iterator; public: SPROUT_STATIC_CONSTEXPR size_type static_size = N; private: template static SPROUT_CONSTEXPR value_type const& dummy_get(value_type const& value) { return value; } template static SPROUT_CONSTEXPR array fill_impl(value_type const& value, sprout::index_tuple) { return array{{dummy_get(value)...}}; } public: value_type elems[static_size ? static_size : 1]; private: template SPROUT_CONSTEXPR std::array to_std_array(sprout::index_tuple) const SPROUT_NOEXCEPT_IF(sprout::is_nothrow_copy_constructible::type>::value) { return std::array{{elems[Indexes]...}}; } public: // construct/copy/destroy: template SPROUT_CXX14_CONSTEXPR array& operator=(array const& rhs) { sprout::copy(rhs.begin(), rhs.end(), begin()); return *this; } template SPROUT_CXX14_CONSTEXPR array& operator=(array&& rhs) { sprout::move(rhs.begin(), rhs.end(), begin()); return *this; } // modifiers: SPROUT_CXX14_CONSTEXPR void fill(const_reference value) { sprout::fill_n(begin(), size(), value); } SPROUT_CONSTEXPR array fill(const_reference value) const { return fill_impl(value, sprout::index_n<0, N>::make()); } SPROUT_CXX14_CONSTEXPR void assign(const_reference value) { fill(value); } SPROUT_CONSTEXPR array assign(const_reference value) const { return fill(value); } SPROUT_CXX14_CONSTEXPR void swap(array& other) SPROUT_NOEXCEPT_IF_EXPR(sprout::swap(std::declval(), std::declval())) { sprout::swap_ranges(other.begin(), other.end(), begin()); } // iterators: #if SPROUT_USE_INDEX_ITERATOR_IMPLEMENTATION SPROUT_CXX14_CONSTEXPR iterator begin() SPROUT_NOEXCEPT { return iterator(*this, 0); } SPROUT_CONSTEXPR const_iterator begin() const SPROUT_NOEXCEPT { return const_iterator(*this, 0); } SPROUT_CXX14_CONSTEXPR iterator end() SPROUT_NOEXCEPT { return iterator(*this, size()); } SPROUT_CONSTEXPR const_iterator end() const SPROUT_NOEXCEPT { return const_iterator(*this, size()); } #else SPROUT_CXX14_CONSTEXPR iterator begin() SPROUT_NOEXCEPT { return iterator(elems); } SPROUT_CONSTEXPR const_iterator begin() const SPROUT_NOEXCEPT { return iterator(elems); } SPROUT_CXX14_CONSTEXPR iterator end() SPROUT_NOEXCEPT { return iterator(elems) + size(); } SPROUT_CONSTEXPR const_iterator end() const SPROUT_NOEXCEPT { return iterator(elems) + size(); } #endif SPROUT_CXX14_CONSTEXPR reverse_iterator rbegin() SPROUT_NOEXCEPT { return reverse_iterator(end()); } SPROUT_CONSTEXPR const_reverse_iterator rbegin() const SPROUT_NOEXCEPT { return const_reverse_iterator(end()); } SPROUT_CXX14_CONSTEXPR reverse_iterator rend() SPROUT_NOEXCEPT { return reverse_iterator(begin()); } SPROUT_CONSTEXPR const_reverse_iterator rend() const SPROUT_NOEXCEPT { return const_reverse_iterator(begin()); } #if SPROUT_USE_INDEX_ITERATOR_IMPLEMENTATION SPROUT_CONSTEXPR const_iterator cbegin() const SPROUT_NOEXCEPT { return const_iterator(*this, 0); } SPROUT_CONSTEXPR const_iterator cend() const SPROUT_NOEXCEPT { return const_iterator(*this, size()); } #else SPROUT_CONSTEXPR const_iterator cbegin() const SPROUT_NOEXCEPT { return const_iterator(elems); } SPROUT_CONSTEXPR const_iterator cend() const SPROUT_NOEXCEPT { return const_iterator(elems) + size(); } #endif SPROUT_CONSTEXPR const_reverse_iterator crbegin() const SPROUT_NOEXCEPT { return const_reverse_iterator(end()); } SPROUT_CONSTEXPR const_reverse_iterator crend() const SPROUT_NOEXCEPT { return const_reverse_iterator(begin()); } // capacity: SPROUT_CONSTEXPR size_type size() const SPROUT_NOEXCEPT { return static_size; } SPROUT_CONSTEXPR size_type max_size() const SPROUT_NOEXCEPT { return size(); } SPROUT_CONSTEXPR bool empty() const SPROUT_NOEXCEPT { return size() == 0; } // element access: SPROUT_CXX14_CONSTEXPR reference operator[](size_type i) { return elems[i]; } SPROUT_CONSTEXPR const_reference operator[](size_type i) const { return elems[i]; } SPROUT_CXX14_CONSTEXPR reference at(size_type i) { return i < size() ? elems[i] : (throw std::out_of_range("array<>: index out of range"), elems[i]) ; } SPROUT_CONSTEXPR const_reference at(size_type i) const { return i < size() ? elems[i] : (throw std::out_of_range("array<>: index out of range"), elems[i]) ; } SPROUT_CXX14_CONSTEXPR reference front() { return elems[0]; } SPROUT_CONSTEXPR const_reference front() const { return elems[0]; } SPROUT_CXX14_CONSTEXPR reference back() { return elems[size() - 1]; } SPROUT_CONSTEXPR const_reference back() const { return elems[size() - 1]; } SPROUT_CXX14_CONSTEXPR pointer data() SPROUT_NOEXCEPT { return elems; } SPROUT_CONSTEXPR const_pointer data() const SPROUT_NOEXCEPT { return elems; } SPROUT_CXX14_CONSTEXPR pointer c_array() SPROUT_NOEXCEPT { return data(); } SPROUT_CONSTEXPR const_pointer c_array() const SPROUT_NOEXCEPT { return data(); } // others: SPROUT_CXX14_CONSTEXPR void rangecheck(size_type i) const { return i >= size() ? throw std::out_of_range("array<>: index out of range") : (void)0 ; } #if SPROUT_USE_INDEX_ITERATOR_IMPLEMENTATION SPROUT_CXX14_CONSTEXPR iterator nth(size_type i) { return i < size() ? iterator(*this, i) : (throw std::out_of_range("array<>: index out of range"), iterator()) ; } SPROUT_CONSTEXPR const_iterator nth(size_type i) const { return i < size() ? const_iterator(*this, i) : (throw std::out_of_range("array<>: index out of range"), const_iterator()) ; } SPROUT_CXX14_CONSTEXPR size_type index_of(iterator p) SPROUT_NOEXCEPT { return p.index(); } SPROUT_CONSTEXPR size_type index_of(const_iterator p) const SPROUT_NOEXCEPT { return p.index(); } #else SPROUT_CXX14_CONSTEXPR iterator nth(size_type i) { return i < size() ? iterator(elems) + i : (throw std::out_of_range("array<>: index out of range"), iterator()) ; } SPROUT_CONSTEXPR const_iterator nth(size_type i) const { return i < size() ? const_iterator(elems) + i : (throw std::out_of_range("array<>: index out of range"), const_iterator()) ; } SPROUT_CXX14_CONSTEXPR size_type index_of(iterator p) SPROUT_NOEXCEPT { return sprout::distance(begin(), p); } SPROUT_CONSTEXPR size_type index_of(const_iterator p) const SPROUT_NOEXCEPT { return sprout::distance(begin(), p); } #endif SPROUT_CONSTEXPR operator std::array() const SPROUT_NOEXCEPT_IF(sprout::is_nothrow_copy_constructible::type>::value) { return to_std_array(sprout::make_index_tuple::make()); } }; template SPROUT_CONSTEXPR_OR_CONST typename sprout::array::size_type sprout::array::static_size; // // swap // template inline SPROUT_CXX14_CONSTEXPR void swap(sprout::array& lhs, sprout::array& rhs) SPROUT_NOEXCEPT_IF_EXPR(lhs.swap(rhs)) { lhs.swap(rhs); } // // to_array // template inline SPROUT_CONSTEXPR sprout::array to_array(sprout::array const& arr) SPROUT_NOEXCEPT_IF(sprout::is_nothrow_copy_constructible::value) { return arr; } namespace detail { template inline SPROUT_CONSTEXPR sprout::array::type, N> to_array_impl(T (& arr)[N], sprout::index_tuple) SPROUT_NOEXCEPT_IF(sprout::is_nothrow_copy_constructible::type>::value) { return sprout::array::type, N>{{arr[Indexes]...}}; } } // namespace detail template inline SPROUT_CONSTEXPR sprout::array::type, N> to_array(T (& arr)[N]) SPROUT_NOEXCEPT_IF(sprout::is_nothrow_copy_constructible::type>::value) { return sprout::detail::to_array_impl(arr, sprout::make_index_tuple::make()); } namespace detail { template inline SPROUT_CONSTEXPR sprout::array to_array_impl(std::array const& arr, sprout::index_tuple) SPROUT_NOEXCEPT_IF(sprout::is_nothrow_copy_constructible::value) { return sprout::array{{arr[Indexes]...}}; } } // namespace detail template inline SPROUT_CONSTEXPR sprout::array to_array(std::array const& arr) SPROUT_NOEXCEPT_IF(sprout::is_nothrow_copy_constructible::value) { return sprout::detail::to_array_impl(arr, sprout::make_index_tuple::make()); } } // namespace sprout #endif // #ifndef SPROUT_ARRAY_ARRAY_HPP