mirror of
https://github.com/bolero-MURAKAMI/Sprout.git
synced 2025-01-13 19:56:43 +00:00
522 lines
17 KiB
C++
522 lines
17 KiB
C++
/*=============================================================================
|
|
Copyright (c) 2011-2017 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_FORWARD_CLIST_HPP
|
|
#define SPROUT_FORWARD_CLIST_HPP
|
|
|
|
#include <iterator>
|
|
#include <type_traits>
|
|
#include <sprout/config.hpp>
|
|
#include <sprout/workaround/std/cstddef.hpp>
|
|
#include <sprout/limits.hpp>
|
|
#include <sprout/memory/addressof.hpp>
|
|
#include <sprout/utility/move.hpp>
|
|
#include <sprout/utility/swap.hpp>
|
|
#include <sprout/utility/value_holder/value_holder.hpp>
|
|
#include <sprout/utility/value_holder/get.hpp>
|
|
#include <sprout/iterator/iterator.hpp>
|
|
|
|
namespace sprout {
|
|
//
|
|
// forward_clist
|
|
//
|
|
template<typename T>
|
|
class forward_clist;
|
|
|
|
namespace detail {
|
|
template<typename List>
|
|
class forward_item_iterator
|
|
: public sprout::iterator<
|
|
std::forward_iterator_tag,
|
|
typename List::value_type,
|
|
typename List::difference_type,
|
|
typename std::conditional<
|
|
std::is_const<typename std::remove_reference<List>::type>::value,
|
|
typename List::const_pointer,
|
|
typename List::pointer
|
|
>::type,
|
|
typename std::conditional<
|
|
std::is_const<typename std::remove_reference<List>::type>::value,
|
|
typename List::const_reference,
|
|
typename List::reference
|
|
>::type
|
|
>
|
|
{
|
|
template<typename T>
|
|
friend class sprout::forward_clist;
|
|
template<typename L>
|
|
friend class sprout::detail::forward_item_iterator;
|
|
public:
|
|
typedef List list_type;
|
|
typedef typename std::conditional<
|
|
std::is_reference<list_type>::value,
|
|
typename std::remove_reference<list_type>::type const&,
|
|
typename std::remove_reference<list_type>::type const
|
|
>::type const_list_type;
|
|
private:
|
|
typedef sprout::iterator<
|
|
std::forward_iterator_tag,
|
|
typename List::value_type,
|
|
typename List::difference_type,
|
|
typename std::conditional<
|
|
std::is_const<typename std::remove_reference<List>::type>::value,
|
|
typename List::const_pointer,
|
|
typename List::pointer
|
|
>::type,
|
|
typename std::conditional<
|
|
std::is_const<typename std::remove_reference<List>::type>::value,
|
|
typename List::const_reference,
|
|
typename List::reference
|
|
>::type
|
|
> base_type;
|
|
typedef typename std::remove_reference<list_type>::type::item_holder_type item_holder_type;
|
|
public:
|
|
typedef typename base_type::iterator_category iterator_category;
|
|
typedef typename base_type::value_type value_type;
|
|
typedef typename base_type::difference_type difference_type;
|
|
typedef typename base_type::pointer pointer;
|
|
typedef typename base_type::reference reference;
|
|
private:
|
|
item_holder_type item;
|
|
private:
|
|
explicit SPROUT_CONSTEXPR forward_item_iterator(item_holder_type const& p)
|
|
: item(p)
|
|
{}
|
|
public:
|
|
SPROUT_CONSTEXPR forward_item_iterator()
|
|
: item()
|
|
{}
|
|
forward_item_iterator(forward_item_iterator const&) = default;
|
|
SPROUT_CONSTEXPR operator forward_item_iterator<const_list_type>() const {
|
|
return forward_item_iterator<const_list_type>(item);
|
|
}
|
|
SPROUT_CONSTEXPR forward_item_iterator next() const {
|
|
return forward_item_iterator(item->next);
|
|
}
|
|
SPROUT_CXX14_CONSTEXPR void swap(forward_item_iterator& other)
|
|
SPROUT_NOEXCEPT_IF_EXPR(sprout::swap(item, other.item))
|
|
{
|
|
sprout::swap(item, other.item);
|
|
}
|
|
SPROUT_CONSTEXPR reference operator*() const {
|
|
return item->get();
|
|
}
|
|
SPROUT_CONSTEXPR pointer operator->() const {
|
|
return item->get_pointer();
|
|
}
|
|
SPROUT_CXX14_CONSTEXPR forward_item_iterator& operator++() {
|
|
forward_item_iterator temp(next());
|
|
temp.swap(*this);
|
|
return *this;
|
|
}
|
|
SPROUT_CXX14_CONSTEXPR forward_item_iterator operator++(int) {
|
|
forward_item_iterator result(*this);
|
|
++*this;
|
|
return result;
|
|
}
|
|
SPROUT_CONSTEXPR bool is_initialized() const SPROUT_NOEXCEPT {
|
|
return item.is_initialized();
|
|
}
|
|
};
|
|
|
|
template<typename List1, typename List2>
|
|
inline SPROUT_CONSTEXPR typename std::enable_if<
|
|
std::is_same<typename std::decay<List1>::type, typename std::decay<List2>::type>::value,
|
|
bool
|
|
>::type
|
|
operator==(sprout::detail::forward_item_iterator<List1> const& lhs, sprout::detail::forward_item_iterator<List2> const& rhs) {
|
|
return !lhs.is_initialized() ? !rhs.is_initialized()
|
|
: rhs.is_initialized() && sprout::addressof(*lhs) == sprout::addressof(*rhs)
|
|
;
|
|
}
|
|
template<typename List1, typename List2>
|
|
inline SPROUT_CONSTEXPR typename std::enable_if<
|
|
std::is_same<typename std::decay<List1>::type, typename std::decay<List2>::type>::value,
|
|
bool
|
|
>::type
|
|
operator!=(sprout::detail::forward_item_iterator<List1> const& lhs, sprout::detail::forward_item_iterator<List2> const& rhs) {
|
|
return !(lhs == rhs);
|
|
}
|
|
|
|
template<typename List>
|
|
inline SPROUT_CXX14_CONSTEXPR void
|
|
swap(sprout::detail::forward_item_iterator<List>& lhs, sprout::detail::forward_item_iterator<List>& rhs)
|
|
SPROUT_NOEXCEPT_IF_EXPR(lhs.swap(rhs))
|
|
{
|
|
lhs.swap(rhs);
|
|
}
|
|
|
|
template<typename List>
|
|
inline SPROUT_CONSTEXPR sprout::detail::forward_item_iterator<List>
|
|
iterator_next(sprout::detail::forward_item_iterator<List> const& it) {
|
|
return it.next();
|
|
}
|
|
} // namespace detail
|
|
|
|
//
|
|
// forward_clist
|
|
//
|
|
template<typename T>
|
|
class forward_clist {
|
|
template<typename List>
|
|
friend class sprout::detail::forward_item_iterator;
|
|
public:
|
|
// types:
|
|
typedef T value_type;
|
|
typedef value_type& reference;
|
|
typedef value_type const& const_reference;
|
|
typedef std::size_t size_type;
|
|
typedef std::ptrdiff_t difference_type;
|
|
typedef value_type* pointer;
|
|
typedef value_type const* const_pointer;
|
|
typedef sprout::detail::forward_item_iterator<forward_clist> iterator;
|
|
typedef sprout::detail::forward_item_iterator<forward_clist const> const_iterator;
|
|
public:
|
|
class item;
|
|
private:
|
|
typedef sprout::value_holder<item&> item_holder_type;
|
|
public:
|
|
//
|
|
// item
|
|
//
|
|
class item {
|
|
friend class forward_clist;
|
|
template<typename List>
|
|
friend class sprout::detail::forward_item_iterator;
|
|
private:
|
|
typedef sprout::value_holder<typename forward_clist::value_type> holder_type;
|
|
typedef sprout::value_holder<item&> item_holder_type;
|
|
public:
|
|
typedef typename holder_type::value_type value_type;
|
|
typedef typename holder_type::lvalue_reference lvalue_reference;
|
|
typedef typename holder_type::rvalue_reference rvalue_reference;
|
|
typedef typename holder_type::reference reference;
|
|
typedef typename holder_type::const_lvalue_reference const_lvalue_reference;
|
|
typedef typename holder_type::const_rvalue_reference const_rvalue_reference;
|
|
typedef typename holder_type::const_reference const_reference;
|
|
typedef typename holder_type::pointer pointer;
|
|
typedef typename holder_type::const_pointer const_pointer;
|
|
typedef typename holder_type::lvalue_reference_type lvalue_reference_type;
|
|
typedef typename holder_type::rvalue_reference_type rvalue_reference_type;
|
|
typedef typename holder_type::reference_type reference_type;
|
|
typedef typename holder_type::reference_const_type reference_const_type;
|
|
typedef typename holder_type::pointer_type pointer_type;
|
|
typedef typename holder_type::pointer_const_type pointer_const_type;
|
|
public:
|
|
static SPROUT_CONSTEXPR reference_type get(item& t) SPROUT_NOEXCEPT {
|
|
return sprout::get(t.val);
|
|
}
|
|
static SPROUT_CONSTEXPR rvalue_reference_type get(item&& t) SPROUT_NOEXCEPT {
|
|
return static_cast<rvalue_reference_type>(get(t));
|
|
}
|
|
static SPROUT_CONSTEXPR reference_const_type get(item const& t) SPROUT_NOEXCEPT {
|
|
return sprout::get(t.val);
|
|
}
|
|
static SPROUT_CONSTEXPR pointer_type get_pointer(item& t) SPROUT_NOEXCEPT {
|
|
return sprout::get_pointer(t.val);
|
|
}
|
|
static SPROUT_CONSTEXPR pointer_type get_pointer(item&& t) SPROUT_NOEXCEPT {
|
|
return get_pointer(t);
|
|
}
|
|
static SPROUT_CONSTEXPR pointer_const_type get_pointer(item const& t) SPROUT_NOEXCEPT {
|
|
return sprout::get_pointer(t.val);
|
|
}
|
|
private:
|
|
holder_type val;
|
|
item_holder_type next;
|
|
private:
|
|
item& operator=(item const&) SPROUT_DELETED_FUNCTION_DECL
|
|
item& operator=(item&&) SPROUT_DELETED_FUNCTION_DECL
|
|
private:
|
|
SPROUT_CONSTEXPR item(typename holder_type::argument_type p, item_holder_type const& n)
|
|
: val(p)
|
|
, next(n)
|
|
{}
|
|
SPROUT_CONSTEXPR item(typename holder_type::movable_argument_type p, item_holder_type const& n)
|
|
: val(sprout::move(p))
|
|
, next(n)
|
|
{}
|
|
|
|
SPROUT_CXX14_CONSTEXPR void swap(item& other)
|
|
SPROUT_NOEXCEPT_IF(
|
|
SPROUT_NOEXCEPT_EXPR(sprout::swap(val, other.val))
|
|
&& SPROUT_NOEXCEPT_EXPR(sprout::swap(next, other.next))
|
|
)
|
|
{
|
|
sprout::swap(val, other.val);
|
|
sprout::swap(next, other.next);
|
|
}
|
|
|
|
SPROUT_CXX14_CONSTEXPR void unlink_all() {
|
|
if (next.is_initialized()) {
|
|
next->unlink_all();
|
|
{
|
|
item_holder_type temp = item_holder_type();
|
|
temp.swap(next);
|
|
}
|
|
}
|
|
}
|
|
SPROUT_CXX14_CONSTEXPR void unlink() {
|
|
item_holder_type temp = item_holder_type();
|
|
temp.swap(next);
|
|
}
|
|
public:
|
|
SPROUT_CONSTEXPR item() SPROUT_NOEXCEPT
|
|
: val(), next()
|
|
{}
|
|
item(item const&) = default;
|
|
SPROUT_CONSTEXPR item(typename holder_type::argument_type p)
|
|
: val(p), next()
|
|
{}
|
|
SPROUT_CONSTEXPR item(typename holder_type::movable_argument_type p)
|
|
: val(sprout::move(p)), next()
|
|
{}
|
|
|
|
SPROUT_CXX14_CONSTEXPR item& operator=(typename holder_type::argument_type p) {
|
|
item temp(p, next);
|
|
temp.swap(p);
|
|
return *this;
|
|
}
|
|
SPROUT_CXX14_CONSTEXPR item& operator=(typename holder_type::movable_argument_type p) {
|
|
item temp(sprout::move(p), next);
|
|
temp.swap(p);
|
|
return *this;
|
|
}
|
|
|
|
SPROUT_CXX14_CONSTEXPR pointer_type operator->() {
|
|
return get_pointer();
|
|
}
|
|
SPROUT_CONSTEXPR pointer_const_type operator->() const {
|
|
return get_pointer();
|
|
}
|
|
SPROUT_CXX14_CONSTEXPR pointer_type get_pointer() {
|
|
return val.get_pointer();
|
|
}
|
|
SPROUT_CONSTEXPR pointer_const_type get_pointer() const {
|
|
return val.get_pointer();
|
|
}
|
|
SPROUT_CXX14_CONSTEXPR reference_type operator*() {
|
|
return get();
|
|
}
|
|
SPROUT_CONSTEXPR reference_const_type operator*() const {
|
|
return get();
|
|
}
|
|
SPROUT_CXX14_CONSTEXPR reference_type get() {
|
|
return val.get();
|
|
}
|
|
SPROUT_CONSTEXPR reference_const_type get() const {
|
|
return val.get();
|
|
}
|
|
SPROUT_CONSTEXPR bool is_linked() const {
|
|
return next.is_initialized();
|
|
}
|
|
};
|
|
private:
|
|
item fst;
|
|
public:
|
|
// construct/copy/destroy:
|
|
SPROUT_CONSTEXPR forward_clist()
|
|
: fst()
|
|
{}
|
|
template<typename InputIterator>
|
|
SPROUT_CXX14_CONSTEXPR forward_clist(InputIterator first, InputIterator last)
|
|
: fst()
|
|
{
|
|
item_holder_type* p = sprout::addressof(fst.next);
|
|
for (; first != last; ++first) {
|
|
*p = *first;
|
|
p = sprout::addressof(*p)->next;
|
|
}
|
|
}
|
|
SPROUT_CXX14_CONSTEXPR forward_clist(forward_clist&& x)
|
|
: fst(sprout::move(x.fst))
|
|
{}
|
|
SPROUT_CXX14_CONSTEXPR forward_clist& operator=(forward_clist&& x) {
|
|
fst = sprout::move(x.fst);
|
|
return *this;
|
|
}
|
|
template<typename InputIterator>
|
|
SPROUT_CXX14_CONSTEXPR void assign(InputIterator first, InputIterator last) {
|
|
forward_clist temp(first, last);
|
|
temp.swap(*this);
|
|
temp.clear();
|
|
}
|
|
// iterators:
|
|
SPROUT_CXX14_CONSTEXPR iterator before_begin() SPROUT_NOEXCEPT {
|
|
return iterator(item_holder_type(fst));
|
|
}
|
|
SPROUT_CONSTEXPR const_iterator before_begin() const SPROUT_NOEXCEPT {
|
|
return const_iterator(item_holder_type(fst));
|
|
}
|
|
SPROUT_CXX14_CONSTEXPR iterator begin() SPROUT_NOEXCEPT {
|
|
return iterator(fst.next);
|
|
}
|
|
SPROUT_CONSTEXPR const_iterator begin() const SPROUT_NOEXCEPT {
|
|
return const_iterator(fst.next);
|
|
}
|
|
SPROUT_CXX14_CONSTEXPR iterator end() SPROUT_NOEXCEPT {
|
|
return iterator();
|
|
}
|
|
SPROUT_CONSTEXPR const_iterator end() const SPROUT_NOEXCEPT {
|
|
return const_iterator();
|
|
}
|
|
SPROUT_CONSTEXPR const_iterator cbegin() const SPROUT_NOEXCEPT {
|
|
return const_iterator(fst.next);
|
|
}
|
|
SPROUT_CONSTEXPR const_iterator cbefore_begin() const SPROUT_NOEXCEPT{
|
|
return const_iterator(fst.next);
|
|
}
|
|
SPROUT_CONSTEXPR const_iterator cend() const SPROUT_NOEXCEPT {
|
|
return const_iterator();
|
|
}
|
|
// capacity:
|
|
SPROUT_CONSTEXPR bool empty() const SPROUT_NOEXCEPT {
|
|
return !!fst.next;
|
|
}
|
|
SPROUT_CONSTEXPR size_type max_size() const SPROUT_NOEXCEPT {
|
|
return sprout::numeric_limits<size_type>::max();
|
|
}
|
|
// element access:
|
|
SPROUT_CXX14_CONSTEXPR reference front() {
|
|
return fst.next->get();
|
|
}
|
|
SPROUT_CONSTEXPR const_reference front() const {
|
|
return fst.next->get();
|
|
}
|
|
// modifiers:
|
|
SPROUT_CXX14_CONSTEXPR void push_front(item& x) {
|
|
item_holder_type nxt(x);
|
|
nxt->next = fst.next;
|
|
fst.next = nxt;
|
|
}
|
|
template<typename InputIterator>
|
|
SPROUT_CXX14_CONSTEXPR void push_front(InputIterator first, InputIterator last) {
|
|
item_holder_type nxt(fst.next);
|
|
item_holder_type* p = sprout::addressof(fst.next);
|
|
for (; first != last; ++first) {
|
|
*p = *first;
|
|
p = sprout::addressof(*p)->next;
|
|
}
|
|
*p = nxt;
|
|
}
|
|
SPROUT_CXX14_CONSTEXPR void pop_front() {
|
|
item_holder_type nxt(fst.next);
|
|
fst.next = fst.next->next;
|
|
nxt->unlink();
|
|
}
|
|
|
|
SPROUT_CXX14_CONSTEXPR iterator insert_after(const_iterator position, item& x) {
|
|
item_holder_type nxt(x);
|
|
nxt->next = position.item->next;
|
|
position.item->next = nxt;
|
|
return iterator(nxt);
|
|
}
|
|
template<typename InputIterator>
|
|
SPROUT_CXX14_CONSTEXPR iterator insert_after(const_iterator position, InputIterator first, InputIterator last) {
|
|
item_holder_type nxt(position.item->next);
|
|
item_holder_type pos(position.item);
|
|
item_holder_type* p = sprout::addressof(pos);
|
|
for (; first != last; ++first) {
|
|
(*p)->next = *first;
|
|
p = sprout::addressof(*p)->next;
|
|
}
|
|
(*p)->next = nxt;
|
|
return iterator(*p);
|
|
}
|
|
|
|
SPROUT_CXX14_CONSTEXPR iterator erase_after(const_iterator position) {
|
|
const_iterator first(position.next());
|
|
position.item->next = first.item->next;
|
|
first.item->unlink();
|
|
return iterator(position.item->next);
|
|
}
|
|
SPROUT_CXX14_CONSTEXPR iterator erase_after(const_iterator position, const_iterator last) {
|
|
const_iterator first(position.next());
|
|
position.item->next = last.item;
|
|
while (first != last) {
|
|
const_iterator nxt(first.next());
|
|
first.item->unlink();
|
|
first = nxt;
|
|
}
|
|
return iterator(last.item);
|
|
}
|
|
|
|
SPROUT_CXX14_CONSTEXPR void swap(forward_clist& other) {
|
|
sprout::swap(fst, other.fst);
|
|
}
|
|
SPROUT_CXX14_CONSTEXPR void clear() SPROUT_NOEXCEPT {
|
|
fst.unlink_all();
|
|
}
|
|
|
|
SPROUT_CXX14_CONSTEXPR void unlink(item& x) {
|
|
for (iterator first = before_begin(), last = end(); first != last; ++first) {
|
|
iterator nxt = first.next();
|
|
if (nxt.item.get_pointer() == sprout::addressof(x)) {
|
|
first.item->next = sprout::move(nxt.item->next);
|
|
nxt.item->unlink();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
template<typename InputIterator>
|
|
SPROUT_CXX14_CONSTEXPR void unlink(InputIterator first, InputIterator last) {
|
|
for (; first != last; ++first) {
|
|
unlink(*first);
|
|
}
|
|
}
|
|
};
|
|
|
|
//
|
|
// get
|
|
//
|
|
template<typename T>
|
|
inline SPROUT_CONSTEXPR typename sprout::forward_clist<T>::item::reference_type
|
|
get(typename sprout::forward_clist<T>::item& x) {
|
|
return sprout::forward_clist<T>::item::get(x);
|
|
}
|
|
template<typename T>
|
|
inline SPROUT_CONSTEXPR typename sprout::forward_clist<T>::item::rvalue_reference
|
|
get(typename sprout::forward_clist<T>::item&& x) {
|
|
return sprout::forward_clist<T>::item::get(sprout::move(x));
|
|
}
|
|
template<typename T>
|
|
inline SPROUT_CONSTEXPR typename sprout::forward_clist<T>::item::reference_const_type
|
|
get(typename sprout::forward_clist<T>::item const& x) {
|
|
return sprout::forward_clist<T>::item::get(x);
|
|
}
|
|
template<typename T>
|
|
inline SPROUT_CONSTEXPR typename sprout::forward_clist<T>::item::pointer_type
|
|
get(typename sprout::forward_clist<T>::item* x) {
|
|
return sprout::forward_clist<T>::item::get_pointer(*x);
|
|
}
|
|
template<typename T>
|
|
inline SPROUT_CONSTEXPR typename sprout::forward_clist<T>::item::pointer_const_type
|
|
get(typename sprout::forward_clist<T>::item const* x) {
|
|
return sprout::forward_clist<T>::item::get_pointer(*x);
|
|
}
|
|
|
|
//
|
|
// get_pointer
|
|
//
|
|
template<typename T>
|
|
inline SPROUT_CONSTEXPR typename sprout::forward_clist<T>::item::pointer_type
|
|
get_pointer(typename sprout::forward_clist<T>::item& x) {
|
|
return sprout::forward_clist<T>::item::get_pointer(x);
|
|
}
|
|
template<typename T>
|
|
inline SPROUT_CONSTEXPR typename sprout::forward_clist<T>::item::pointer_type
|
|
get_pointer(typename sprout::forward_clist<T>::item&& x) {
|
|
return sprout::forward_clist<T>::item::get_pointer(sprout::move(x));
|
|
}
|
|
template<typename T>
|
|
inline SPROUT_CONSTEXPR typename sprout::forward_clist<T>::item::pointer_const_type
|
|
get_pointer(typename sprout::forward_clist<T>::item const& x) {
|
|
return sprout::forward_clist<T>::item::get_pointer(x);
|
|
}
|
|
} // namespace sprout
|
|
|
|
#endif // #ifndef SPROUT_FORWARD_CLIST_HPP
|