// Copyright 2019 Hans Dembinski // // 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 BOOST_HISTOGRAM_DETAIL_SUB_ARRAY_HPP #define BOOST_HISTOGRAM_DETAIL_SUB_ARRAY_HPP #include #include #include namespace boost { namespace histogram { namespace detail { // Like std::array, but allows to use less than maximum capacity. // Cannot inherit from std::array, since this confuses span. template class sub_array { static constexpr bool swap_element_is_noexcept() noexcept { using std::swap; return noexcept(swap(std::declval(), std::declval())); } public: using element_type = T; using size_type = std::size_t; using reference = T&; using const_reference = const T&; using pointer = T*; using const_pointer = const T*; using iterator = pointer; using const_iterator = const_pointer; sub_array() = default; explicit sub_array(std::size_t s) noexcept : size_(s) { assert(size_ <= N); } sub_array(std::size_t s, const T& value) noexcept( std::is_nothrow_assignable::value) : sub_array(s) { fill(value); } sub_array(std::initializer_list il) noexcept( std::is_nothrow_assignable::value) : sub_array(il.size()) { std::copy(il.begin(), il.end(), data_); } reference at(size_type pos) noexcept { if (pos >= size()) BOOST_THROW_EXCEPTION(std::out_of_range{"pos is out of range"}); return data_[pos]; } const_reference at(size_type pos) const noexcept { if (pos >= size()) BOOST_THROW_EXCEPTION(std::out_of_range{"pos is out of range"}); return data_[pos]; } reference operator[](size_type pos) noexcept { return data_[pos]; } const_reference operator[](size_type pos) const noexcept { return data_[pos]; } reference front() noexcept { return data_[0]; } const_reference front() const noexcept { return data_[0]; } reference back() noexcept { return data_[size_ - 1]; } const_reference back() const noexcept { return data_[size_ - 1]; } pointer data() noexcept { return static_cast(data_); } const_pointer data() const noexcept { return static_cast(data_); } iterator begin() noexcept { return data_; } const_iterator begin() const noexcept { return data_; } iterator end() noexcept { return begin() + size_; } const_iterator end() const noexcept { return begin() + size_; } const_iterator cbegin() const noexcept { return data_; } const_iterator cend() const noexcept { return cbegin() + size_; } constexpr size_type max_size() const noexcept { return N; } size_type size() const noexcept { return size_; } bool empty() const noexcept { return size_ == 0; } void fill(const_reference value) noexcept( std::is_nothrow_assignable::value) { std::fill(begin(), end(), value); } void swap(sub_array& other) noexcept(swap_element_is_noexcept()) { using std::swap; const size_type s = (std::max)(size(), other.size()); for (auto i = begin(), j = other.begin(), end = begin() + s; i != end; ++i, ++j) swap(*i, *j); swap(size_, other.size_); } private: size_type size_ = 0; element_type data_[N]; }; template bool operator==(const sub_array& a, const sub_array& b) noexcept { return std::equal(a.begin(), a.end(), b.begin(), b.end()); } template bool operator!=(const sub_array& a, const sub_array& b) noexcept { return !(a == b); } } // namespace detail } // namespace histogram } // namespace boost namespace std { template void swap(::boost::histogram::detail::sub_array& a, ::boost::histogram::detail::sub_array& b) noexcept(noexcept(a.swap(b))) { a.swap(b); } } // namespace std #endif