rnd_index_ops.hpp

来自「Boost provides free peer-reviewed portab」· HPP 代码 · 共 207 行

HPP
207
字号
/* Copyright 2003-2008 Joaquin M Lopez Munoz. * 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) * * See http://www.boost.org/libs/multi_index for library home page. */#ifndef BOOST_MULTI_INDEX_DETAIL_RND_INDEX_OPS_HPP#define BOOST_MULTI_INDEX_DETAIL_RND_INDEX_OPS_HPP#if defined(_MSC_VER)&&(_MSC_VER>=1200)#pragma once#endif#include <boost/config.hpp> /* keep it first to prevent nasty warns in MSVC */#include <algorithm>#include <boost/multi_index/detail/rnd_index_ptr_array.hpp>#include <functional>namespace boost{namespace multi_index{namespace detail{/* Common code for random_access_index memfuns having templatized and * non-templatized versions. */template<typename Node,typename Allocator,typename Predicate>Node* random_access_index_remove(  random_access_index_ptr_array<Allocator>& ptrs,Predicate pred  BOOST_APPEND_EXPLICIT_TEMPLATE_TYPE(Node)){  typedef typename Node::value_type value_type;  typedef typename Node::impl_ptr_pointer impl_ptr_pointer;  impl_ptr_pointer first=ptrs.begin(),                   res=first,                   last=ptrs.end();  for(;first!=last;++first){    if(!pred(        const_cast<const value_type&>(Node::from_impl(*first)->value()))){      if(first!=res){        std::swap(*first,*res);        (*first)->up()=first;        (*res)->up()=res;      }      ++res;    }  }  return Node::from_impl(*res);}template<typename Node,typename Allocator,class BinaryPredicate>Node* random_access_index_unique(  random_access_index_ptr_array<Allocator>& ptrs,BinaryPredicate binary_pred  BOOST_APPEND_EXPLICIT_TEMPLATE_TYPE(Node)){  typedef typename Node::value_type       value_type;  typedef typename Node::impl_ptr_pointer impl_ptr_pointer;  impl_ptr_pointer first=ptrs.begin(),                   res=first,                   last=ptrs.end();  if(first!=last){    for(;++first!=last;){      if(!binary_pred(           const_cast<const value_type&>(Node::from_impl(*res)->value()),           const_cast<const value_type&>(Node::from_impl(*first)->value()))){        ++res;        if(first!=res){          std::swap(*first,*res);          (*first)->up()=first;          (*res)->up()=res;        }      }    }    ++res;  }  return Node::from_impl(*res);}template<typename Node,typename Allocator,typename Compare>void random_access_index_inplace_merge(  const Allocator& al,  random_access_index_ptr_array<Allocator>& ptrs,  BOOST_DEDUCED_TYPENAME Node::impl_ptr_pointer first1,Compare comp  BOOST_APPEND_EXPLICIT_TEMPLATE_TYPE(Node)){  typedef typename Node::value_type       value_type;  typedef typename Node::impl_pointer     impl_pointer;  typedef typename Node::impl_ptr_pointer impl_ptr_pointer;  auto_space<impl_pointer,Allocator> spc(al,ptrs.size());  impl_ptr_pointer first0=ptrs.begin(),                   last0=first1,                   last1=ptrs.end(),                   out=spc.data();  while(first0!=last0&&first1!=last1){    if(comp(        const_cast<const value_type&>(Node::from_impl(*first1)->value()),        const_cast<const value_type&>(Node::from_impl(*first0)->value()))){      *out++=*first1++;    }    else{      *out++=*first0++;    }  }  std::copy(&*first0,&*last0,&*out);  std::copy(&*first1,&*last1,&*out);  first1=ptrs.begin();  out=spc.data();  while(first1!=last1){    *first1=*out++;    (*first1)->up()=first1;    ++first1;  }}/* sorting *//* auxiliary stuff */template<typename Node,typename Compare>struct random_access_index_sort_compare:  std::binary_function<    typename Node::impl_pointer,    typename Node::impl_pointer,bool>{  random_access_index_sort_compare(Compare comp_=Compare()):comp(comp_){}  bool operator()(    typename Node::impl_pointer x,typename Node::impl_pointer y)const  {    typedef typename Node::value_type value_type;    return comp(      const_cast<const value_type&>(Node::from_impl(x)->value()),      const_cast<const value_type&>(Node::from_impl(y)->value()));  }private:  Compare comp;};template<typename Node,typename Allocator,class Compare>void random_access_index_sort(  const Allocator& al,  random_access_index_ptr_array<Allocator>& ptrs,  Compare comp  BOOST_APPEND_EXPLICIT_TEMPLATE_TYPE(Node)){  /* The implementation is extremely simple: an auxiliary   * array of pointers is sorted using stdlib facilities and   * then used to rearrange the index. This is suboptimal   * in space and time, but has some advantages over other   * possible approaches:   *   - Use std::stable_sort() directly on ptrs using some   *     special iterator in charge of maintaining pointers   *     and up() pointers in sync: we cannot guarantee   *     preservation of the container invariants in the face of   *     exceptions, if, for instance, std::stable_sort throws   *     when ptrs transitorily contains duplicate elements.   *   - Rewrite the internal algorithms of std::stable_sort   *     adapted for this case: besides being a fair amount of   *     work, making a stable sort compatible with Boost.MultiIndex   *     invariants (basically, no duplicates or missing elements   *     even if an exception is thrown) is complicated, error-prone   *     and possibly won't perform much better than the   *     solution adopted.   */  typedef typename Node::value_type         value_type;  typedef typename Node::impl_pointer       impl_pointer;  typedef typename Node::impl_ptr_pointer   impl_ptr_pointer;  typedef random_access_index_sort_compare<    Node,Compare>                           ptr_compare;    impl_ptr_pointer   first=ptrs.begin();  impl_ptr_pointer   last=ptrs.end();  auto_space<    impl_pointer,    Allocator>       spc(al,ptrs.size());  impl_ptr_pointer   buf=spc.data();  std::copy(&*first,&*last,&*buf);  std::stable_sort(&*buf,&*buf+ptrs.size(),ptr_compare(comp));  while(first!=last){    *first=*buf++;    (*first)->up()=first;    ++first;  }}} /* namespace multi_index::detail */} /* namespace multi_index */} /* namespace boost */#endif

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?