#ifndef SPROUT_ARRAY_HPP #define SPROUT_ARRAY_HPP #include #include #include #include #include #include #include #include #include #include #include #include HDR_ALGORITHM_SSCRISK_CEL_OR_SPROUT_DETAIL #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; SPROUT_STATIC_CONSTEXPR size_type fixed_size = static_size; public: value_type elems[static_size ? static_size : 1]; public: void fill(const_reference value) { std::fill_n(begin(), size(), value); } void swap(array& other) SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(std::swap(std::declval(), std::declval()))) { std::swap_ranges(other.begin(), other.end(), begin()); } // iterators: #if SPROUT_USE_INDEX_ITERATOR_IMPLEMENTATION iterator begin() SPROUT_NOEXCEPT { return iterator(*this, 0); } SPROUT_CONSTEXPR const_iterator begin() const SPROUT_NOEXCEPT { return const_iterator(*this, 0); } iterator end() SPROUT_NOEXCEPT { return iterator(*this, size()); } SPROUT_CONSTEXPR const_iterator end() const SPROUT_NOEXCEPT { return const_iterator(*this, size()); } #else iterator begin() SPROUT_NOEXCEPT { return &elems[0]; } SPROUT_CONSTEXPR const_iterator begin() const SPROUT_NOEXCEPT { return &elems[0]; } iterator end() SPROUT_NOEXCEPT { return &elems[0] + size(); } SPROUT_CONSTEXPR const_iterator end() const SPROUT_NOEXCEPT { return &elems[0] + size(); } #endif reverse_iterator rbegin() SPROUT_NOEXCEPT { return reverse_iterator(end()); } SPROUT_CONSTEXPR const_reverse_iterator rbegin() const SPROUT_NOEXCEPT { return const_reverse_iterator(end()); } 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 &elems[0]; } SPROUT_CONSTEXPR const_iterator cend() const SPROUT_NOEXCEPT { return &elems[0] + 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: reference operator[](size_type i) { return elems[i]; } SPROUT_CONSTEXPR const_reference operator[](size_type i) const { return elems[i]; } 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]) ; } reference front() { return elems[0]; } SPROUT_CONSTEXPR const_reference front() const { return elems[0]; } reference back() { return elems[size() - 1]; } SPROUT_CONSTEXPR const_reference back() const { return elems[size() - 1]; } pointer data() SPROUT_NOEXCEPT { return &elems[0]; } SPROUT_CONSTEXPR const_pointer data() const SPROUT_NOEXCEPT { return &elems[0]; } // others: template array& operator=(array const& rhs) { std::copy(rhs.begin(), rhs.end(), begin()); return *this; } template array& operator=(array&& rhs) { std::move(rhs.begin(), rhs.end(), begin()); return *this; } pointer c_array() SPROUT_NOEXCEPT { return &elems[0]; } void assign(const_reference value) { fill(value); } void rangecheck(size_type i) const { if (i >= size()) { throw std::out_of_range("array<>: index out of range"); } } }; template SPROUT_CONSTEXPR typename sprout::array::size_type sprout::array::static_size; template SPROUT_CONSTEXPR typename sprout::array::size_type sprout::array::fixed_size; // // operator== // operator!= // operator< // operator> // operator<= // operator>= // template SPROUT_CONSTEXPR inline bool operator==(sprout::array const& lhs, sprout::array const& rhs) { return NS_SSCRISK_CEL_OR_SPROUT_DETAIL::equal(lhs.begin(), lhs.end(), rhs.begin()); } template SPROUT_CONSTEXPR inline bool operator!=(sprout::array const& lhs, sprout::array const& rhs) { return !(lhs == rhs); } template SPROUT_CONSTEXPR inline bool operator<(sprout::array const& lhs, sprout::array const& rhs) { return NS_SSCRISK_CEL_OR_SPROUT_DETAIL::lexicographical_compare(lhs.begin(), lhs.end(), rhs.begin(), rhs.end()); } template SPROUT_CONSTEXPR inline bool operator>(sprout::array const& lhs, sprout::array const& rhs) { return rhs < lhs; } template SPROUT_CONSTEXPR inline bool operator<=(sprout::array const& lhs, sprout::array const& rhs) { return !(rhs < lhs); } template SPROUT_CONSTEXPR inline bool operator>=(sprout::array const& lhs, sprout::array const& rhs) { return !(lhs < rhs); } // // swap // template inline void swap(sprout::array& lhs, sprout::array& rhs) SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(lhs.swap(rhs))) { lhs.swap(rhs); } // // rebind_fixed_size // template struct rebind_fixed_size > { public: template >::size_type S> struct apply { public: typedef sprout::array type; }; }; // // make_array // template SPROUT_CONSTEXPR inline sprout::array make_array(Types&&... args) { return sprout::array{{sprout::forward(args)...}}; } // // make_common_array // template SPROUT_CONSTEXPR inline sprout::array::type...>::type, sizeof...(Types)> make_common_array(Types&&... args) { return sprout::array::type...>::type, sizeof...(Types)>{{sprout::forward(args)...}}; } namespace detail { template SPROUT_CONSTEXPR inline sprout::array to_array_impl( T const (& arr)[N], sprout::index_tuple ) { return sprout::array{{arr[Indexes]...}}; } } // namespace detail // // to_array // template SPROUT_CONSTEXPR inline sprout::array to_array(T const (& arr)[N]) { return sprout::detail::to_array_impl(arr, typename sprout::index_range<0, N>::type()); } namespace detail { template struct is_array_impl : public std::false_type {}; template struct is_array_impl< T, typename std::enable_if< std::is_same< T, sprout::array >::value >::type > : public std::true_type {}; } // namespace detail // // is_array // template struct is_array : public sprout::detail::is_array_impl {}; } // namespace sprout namespace std { // // tuple_size // template struct tuple_size > { public: typedef std::integral_constant type; SPROUT_STATIC_CONSTEXPR std::size_t value = type::value; }; // // tuple_element // template struct tuple_element > { public: static_assert(I < N, "tuple_element<>: index out of range"); typedef T type; }; } // namespace std #endif // #ifndef SPROUT_ARRAY_HPP