container_traits.hpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630
  1. // (C) Copyright Jeremy Siek 2004
  2. // (C) Copyright Thomas Claveirole 2010
  3. // (C) Copyright Ignacy Gawedzki 2010
  4. // Distributed under the Boost Software License, Version 1.0. (See
  5. // accompanying file LICENSE_1_0.txt or copy at
  6. // http://www.boost.org/LICENSE_1_0.txt)
  7. #ifndef BOOST_GRAPH_DETAIL_CONTAINER_TRAITS_H
  8. #define BOOST_GRAPH_DETAIL_CONTAINER_TRAITS_H
  9. // Sure would be nice to be able to forward declare these
  10. // instead of pulling in all the headers. Too bad that
  11. // is not legal. There ought to be a standard <stlfwd> header. -JGS
  12. #include <boost/next_prior.hpp>
  13. #include <algorithm> // for std::remove
  14. #include <utility>
  15. #include <vector>
  16. #include <list>
  17. #include <map>
  18. #include <set>
  19. #include <boost/unordered_set.hpp>
  20. #include <boost/unordered_map.hpp>
  21. #if !defined BOOST_NO_SLIST
  22. # ifdef BOOST_SLIST_HEADER
  23. # include BOOST_SLIST_HEADER
  24. # else
  25. # include <slist>
  26. # endif
  27. #endif
  28. #ifndef BOOST_NO_CXX11_HDR_UNORDERED_SET
  29. #include <unordered_set>
  30. #endif
  31. #ifndef BOOST_NO_CXX11_HDR_UNORDERED_MAP
  32. #include <unordered_map>
  33. #endif
  34. #ifdef BOOST_NO_CXX11_RVALUE_REFERENCES
  35. #define BOOST_PENDING_FWD_TYPE(type) const type&
  36. #define BOOST_PENDING_FWD_VALUE(type, var) (var)
  37. #else
  38. #define BOOST_PENDING_FWD_TYPE(type) type&&
  39. #define BOOST_PENDING_FWD_VALUE(type, var) (std::forward<type>((var)))
  40. #endif
  41. // The content of this file is in 'graph_detail' because otherwise
  42. // there will be name clashes with
  43. // sandbox/boost/sequence_algo/container_traits.hpp
  44. // The 'detail' subnamespace will still cause problems.
  45. namespace boost { namespace graph_detail {
  46. //======================================================================
  47. // Container Category Tags
  48. //
  49. // They use virtual inheritance because there are lots of
  50. // inheritance diamonds.
  51. struct container_tag { };
  52. struct forward_container_tag : virtual public container_tag { };
  53. struct reversible_container_tag : virtual public forward_container_tag { };
  54. struct random_access_container_tag
  55. : virtual public reversible_container_tag { };
  56. struct sequence_tag : virtual public forward_container_tag { };
  57. struct associative_container_tag : virtual public forward_container_tag { };
  58. struct sorted_associative_container_tag
  59. : virtual public associative_container_tag,
  60. virtual public reversible_container_tag { };
  61. struct front_insertion_sequence_tag : virtual public sequence_tag { };
  62. struct back_insertion_sequence_tag : virtual public sequence_tag { };
  63. struct unique_associative_container_tag
  64. : virtual public associative_container_tag { };
  65. struct multiple_associative_container_tag
  66. : virtual public associative_container_tag { };
  67. struct simple_associative_container_tag
  68. : virtual public associative_container_tag { };
  69. struct pair_associative_container_tag
  70. : virtual public associative_container_tag { };
  71. //======================================================================
  72. // Iterator Stability Tags
  73. //
  74. // Do mutating operations such as insert/erase/resize invalidate all
  75. // outstanding iterators?
  76. struct stable_tag { };
  77. struct unstable_tag { };
  78. //======================================================================
  79. // Container Traits Class and container_category() function
  80. // don't use this unless there is partial specialization
  81. template <class Container>
  82. struct container_traits {
  83. typedef typename Container::category category;
  84. typedef typename Container::iterator_stability iterator_stability;
  85. };
  86. // Use this as a compile-time assertion that X is stable
  87. inline void require_stable(stable_tag) { }
  88. // std::vector
  89. struct vector_tag :
  90. virtual public random_access_container_tag,
  91. virtual public back_insertion_sequence_tag { };
  92. template <class T, class Alloc>
  93. vector_tag container_category(const std::vector<T,Alloc>&)
  94. { return vector_tag(); }
  95. template <class T, class Alloc>
  96. unstable_tag iterator_stability(const std::vector<T,Alloc>&)
  97. { return unstable_tag(); }
  98. template <class T, class Alloc>
  99. struct container_traits< std::vector<T,Alloc> > {
  100. typedef vector_tag category;
  101. typedef unstable_tag iterator_stability;
  102. };
  103. // std::list
  104. struct list_tag :
  105. virtual public reversible_container_tag,
  106. virtual public back_insertion_sequence_tag
  107. // this causes problems for push_dispatch...
  108. // virtual public front_insertion_sequence_tag
  109. { };
  110. template <class T, class Alloc>
  111. list_tag container_category(const std::list<T,Alloc>&)
  112. { return list_tag(); }
  113. template <class T, class Alloc>
  114. stable_tag iterator_stability(const std::list<T,Alloc>&)
  115. { return stable_tag(); }
  116. template <class T, class Alloc>
  117. struct container_traits< std::list<T,Alloc> > {
  118. typedef list_tag category;
  119. typedef stable_tag iterator_stability;
  120. };
  121. // std::slist
  122. #ifndef BOOST_NO_SLIST
  123. template <class T, class Alloc>
  124. struct container_traits<BOOST_STD_EXTENSION_NAMESPACE::slist<T,Alloc> > {
  125. typedef front_insertion_sequence_tag category;
  126. typedef stable_tag iterator_stability;
  127. };
  128. template <class T, class Alloc>
  129. front_insertion_sequence_tag container_category(
  130. const BOOST_STD_EXTENSION_NAMESPACE::slist<T,Alloc>&
  131. )
  132. { return front_insertion_sequence_tag(); }
  133. template <class T, class Alloc>
  134. stable_tag iterator_stability(
  135. const BOOST_STD_EXTENSION_NAMESPACE::slist<T,Alloc>&)
  136. { return stable_tag(); }
  137. #endif
  138. // std::set
  139. struct set_tag :
  140. virtual public sorted_associative_container_tag,
  141. virtual public simple_associative_container_tag,
  142. virtual public unique_associative_container_tag
  143. { };
  144. template <class Key, class Cmp, class Alloc>
  145. set_tag container_category(const std::set<Key,Cmp,Alloc>&)
  146. { return set_tag(); }
  147. template <class Key, class Cmp, class Alloc>
  148. stable_tag iterator_stability(const std::set<Key,Cmp,Alloc>&)
  149. { return stable_tag(); }
  150. template <class Key, class Cmp, class Alloc>
  151. struct container_traits< std::set<Key,Cmp,Alloc> > {
  152. typedef set_tag category;
  153. typedef stable_tag iterator_stability;
  154. };
  155. // std::multiset
  156. struct multiset_tag :
  157. virtual public sorted_associative_container_tag,
  158. virtual public simple_associative_container_tag,
  159. virtual public multiple_associative_container_tag
  160. { };
  161. template <class Key, class Cmp, class Alloc>
  162. multiset_tag container_category(const std::multiset<Key,Cmp,Alloc>&)
  163. { return multiset_tag(); }
  164. template <class Key, class Cmp, class Alloc>
  165. stable_tag iterator_stability(const std::multiset<Key,Cmp,Alloc>&)
  166. { return stable_tag(); }
  167. template <class Key, class Cmp, class Alloc>
  168. struct container_traits< std::multiset<Key,Cmp,Alloc> > {
  169. typedef multiset_tag category;
  170. typedef stable_tag iterator_stability;
  171. };
  172. // deque
  173. // std::map
  174. struct map_tag :
  175. virtual public sorted_associative_container_tag,
  176. virtual public pair_associative_container_tag,
  177. virtual public unique_associative_container_tag
  178. { };
  179. template <class Key, class T, class Cmp, class Alloc>
  180. struct container_traits< std::map<Key,T,Cmp,Alloc> > {
  181. typedef map_tag category;
  182. typedef stable_tag iterator_stability;
  183. };
  184. template <class Key, class T, class Cmp, class Alloc>
  185. map_tag container_category(const std::map<Key,T,Cmp,Alloc>&)
  186. { return map_tag(); }
  187. template <class Key, class T, class Cmp, class Alloc>
  188. stable_tag iterator_stability(const std::map<Key,T,Cmp,Alloc>&)
  189. { return stable_tag(); }
  190. // std::multimap
  191. struct multimap_tag :
  192. virtual public sorted_associative_container_tag,
  193. virtual public pair_associative_container_tag,
  194. virtual public multiple_associative_container_tag
  195. { };
  196. template <class Key, class T, class Cmp, class Alloc>
  197. struct container_traits< std::multimap<Key,T,Cmp,Alloc> > {
  198. typedef multimap_tag category;
  199. typedef stable_tag iterator_stability;
  200. };
  201. template <class Key, class T, class Cmp, class Alloc>
  202. multimap_tag container_category(const std::multimap<Key,T,Cmp,Alloc>&)
  203. { return multimap_tag(); }
  204. template <class Key, class T, class Cmp, class Alloc>
  205. stable_tag iterator_stability(const std::multimap<Key,T,Cmp,Alloc>&)
  206. { return stable_tag(); }
  207. // hash_set, hash_map
  208. struct unordered_set_tag :
  209. virtual public simple_associative_container_tag,
  210. virtual public unique_associative_container_tag
  211. { };
  212. struct unordered_multiset_tag :
  213. virtual public simple_associative_container_tag,
  214. virtual public multiple_associative_container_tag
  215. { };
  216. struct unordered_map_tag :
  217. virtual public pair_associative_container_tag,
  218. virtual public unique_associative_container_tag
  219. { };
  220. struct unordered_multimap_tag :
  221. virtual public pair_associative_container_tag,
  222. virtual public multiple_associative_container_tag
  223. { };
  224. template <class Key, class Eq, class Hash, class Alloc>
  225. struct container_traits< boost::unordered_set<Key,Eq,Hash,Alloc> > {
  226. typedef unordered_set_tag category;
  227. typedef unstable_tag iterator_stability;
  228. };
  229. template <class Key, class T, class Eq, class Hash, class Alloc>
  230. struct container_traits< boost::unordered_map<Key,T,Eq,Hash,Alloc> > {
  231. typedef unordered_map_tag category;
  232. typedef unstable_tag iterator_stability;
  233. };
  234. template <class Key, class Eq, class Hash, class Alloc>
  235. struct container_traits< boost::unordered_multiset<Key,Eq,Hash,Alloc> > {
  236. typedef unordered_multiset_tag category;
  237. typedef unstable_tag iterator_stability;
  238. };
  239. template <class Key, class T, class Eq, class Hash, class Alloc>
  240. struct container_traits< boost::unordered_multimap<Key,T,Eq,Hash,Alloc> > {
  241. typedef unordered_multimap_tag category;
  242. typedef unstable_tag iterator_stability;
  243. };
  244. template <class Key, class Eq, class Hash, class Alloc>
  245. unordered_set_tag
  246. container_category(const boost::unordered_set<Key,Eq,Hash,Alloc>&)
  247. { return unordered_set_tag(); }
  248. template <class Key, class T, class Eq, class Hash, class Alloc>
  249. unordered_map_tag
  250. container_category(const boost::unordered_map<Key,T,Eq,Hash,Alloc>&)
  251. { return unordered_map_tag(); }
  252. template <class Key, class Eq, class Hash, class Alloc>
  253. unstable_tag iterator_stability(const boost::unordered_set<Key,Eq,Hash,Alloc>&)
  254. { return unstable_tag(); }
  255. template <class Key, class T, class Eq, class Hash, class Alloc>
  256. unstable_tag iterator_stability(const boost::unordered_map<Key,T,Eq,Hash,Alloc>&)
  257. { return unstable_tag(); }
  258. template <class Key, class Eq, class Hash, class Alloc>
  259. unordered_multiset_tag
  260. container_category(const boost::unordered_multiset<Key,Eq,Hash,Alloc>&)
  261. { return unordered_multiset_tag(); }
  262. template <class Key, class T, class Eq, class Hash, class Alloc>
  263. unordered_multimap_tag
  264. container_category(const boost::unordered_multimap<Key,T,Eq,Hash,Alloc>&)
  265. { return unordered_multimap_tag(); }
  266. template <class Key, class Eq, class Hash, class Alloc>
  267. unstable_tag
  268. iterator_stability(const boost::unordered_multiset<Key,Eq,Hash,Alloc>&)
  269. { return unstable_tag(); }
  270. template <class Key, class T, class Eq, class Hash, class Alloc>
  271. unstable_tag
  272. iterator_stability(const boost::unordered_multimap<Key,T,Eq,Hash,Alloc>&)
  273. { return unstable_tag(); }
  274. #ifndef BOOST_NO_CXX11_HDR_UNORDERED_SET
  275. template <class Key, class Eq, class Hash, class Alloc>
  276. struct container_traits< std::unordered_set<Key,Eq,Hash,Alloc> > {
  277. typedef unordered_set_tag category;
  278. typedef unstable_tag iterator_stability;
  279. };
  280. #endif
  281. #ifndef BOOST_NO_CXX11_HDR_UNORDERED_MAP
  282. template <class Key, class T, class Eq, class Hash, class Alloc>
  283. struct container_traits< std::unordered_map<Key,T,Eq,Hash,Alloc> > {
  284. typedef unordered_map_tag category;
  285. typedef unstable_tag iterator_stability;
  286. };
  287. #endif
  288. #ifndef BOOST_NO_CXX11_HDR_UNORDERED_SET
  289. template <class Key, class Eq, class Hash, class Alloc>
  290. struct container_traits< std::unordered_multiset<Key,Eq,Hash,Alloc> > {
  291. typedef unordered_multiset_tag category;
  292. typedef unstable_tag iterator_stability;
  293. };
  294. #endif
  295. #ifndef BOOST_NO_CXX11_HDR_UNORDERED_MAP
  296. template <class Key, class T, class Eq, class Hash, class Alloc>
  297. struct container_traits< std::unordered_multimap<Key,T,Eq,Hash,Alloc> > {
  298. typedef unordered_multimap_tag category;
  299. typedef unstable_tag iterator_stability;
  300. };
  301. #endif
  302. #ifndef BOOST_NO_CXX11_HDR_UNORDERED_SET
  303. template <class Key, class Eq, class Hash, class Alloc>
  304. unordered_set_tag
  305. container_category(const std::unordered_set<Key,Eq,Hash,Alloc>&)
  306. { return unordered_set_tag(); }
  307. #endif
  308. #ifndef BOOST_NO_CXX11_HDR_UNORDERED_MAP
  309. template <class Key, class T, class Eq, class Hash, class Alloc>
  310. unordered_map_tag
  311. container_category(const std::unordered_map<Key,T,Eq,Hash,Alloc>&)
  312. { return unordered_map_tag(); }
  313. #endif
  314. #ifndef BOOST_NO_CXX11_HDR_UNORDERED_SET
  315. template <class Key, class Eq, class Hash, class Alloc>
  316. unstable_tag iterator_stability(const std::unordered_set<Key,Eq,Hash,Alloc>&)
  317. { return unstable_tag(); }
  318. #endif
  319. #ifndef BOOST_NO_CXX11_HDR_UNORDERED_MAP
  320. template <class Key, class T, class Eq, class Hash, class Alloc>
  321. unstable_tag iterator_stability(const std::unordered_map<Key,T,Eq,Hash,Alloc>&)
  322. { return unstable_tag(); }
  323. #endif
  324. #ifndef BOOST_NO_CXX11_HDR_UNORDERED_SET
  325. template <class Key, class Eq, class Hash, class Alloc>
  326. unordered_multiset_tag
  327. container_category(const std::unordered_multiset<Key,Eq,Hash,Alloc>&)
  328. { return unordered_multiset_tag(); }
  329. #endif
  330. #ifndef BOOST_NO_CXX11_HDR_UNORDERED_MAP
  331. template <class Key, class T, class Eq, class Hash, class Alloc>
  332. unordered_multimap_tag
  333. container_category(const std::unordered_multimap<Key,T,Eq,Hash,Alloc>&)
  334. { return unordered_multimap_tag(); }
  335. #endif
  336. #ifndef BOOST_NO_CXX11_HDR_UNORDERED_SET
  337. template <class Key, class Eq, class Hash, class Alloc>
  338. unstable_tag
  339. iterator_stability(const std::unordered_multiset<Key,Eq,Hash,Alloc>&)
  340. { return unstable_tag(); }
  341. #endif
  342. #ifndef BOOST_NO_CXX11_HDR_UNORDERED_MAP
  343. template <class Key, class T, class Eq, class Hash, class Alloc>
  344. unstable_tag
  345. iterator_stability(const std::unordered_multimap<Key,T,Eq,Hash,Alloc>&)
  346. { return unstable_tag(); }
  347. #endif
  348. //===========================================================================
  349. // Generalized Container Functions
  350. // Erase
  351. template <class Sequence, class T>
  352. void erase_dispatch(Sequence& c, const T& x,
  353. sequence_tag)
  354. {
  355. c.erase(std::remove(c.begin(), c.end(), x), c.end());
  356. }
  357. template <class AssociativeContainer, class T>
  358. void erase_dispatch(AssociativeContainer& c, const T& x,
  359. associative_container_tag)
  360. {
  361. c.erase(x);
  362. }
  363. template <class Container, class T>
  364. void erase(Container& c, const T& x)
  365. {
  366. erase_dispatch(c, x, container_category(c));
  367. }
  368. // Erase If
  369. template <class Sequence, class Predicate, class IteratorStability>
  370. void erase_if_dispatch(Sequence& c, Predicate p,
  371. sequence_tag, IteratorStability)
  372. {
  373. #if 0
  374. c.erase(std::remove_if(c.begin(), c.end(), p), c.end());
  375. #else
  376. if (! c.empty())
  377. c.erase(std::remove_if(c.begin(), c.end(), p), c.end());
  378. #endif
  379. }
  380. template <class AssociativeContainer, class Predicate>
  381. void erase_if_dispatch(AssociativeContainer& c, Predicate p,
  382. associative_container_tag, stable_tag)
  383. {
  384. typename AssociativeContainer::iterator i, next;
  385. for (i = next = c.begin(); next != c.end(); i = next) {
  386. ++next;
  387. if (p(*i))
  388. c.erase(i);
  389. }
  390. }
  391. template <class AssociativeContainer, class Predicate>
  392. void erase_if_dispatch(AssociativeContainer& c, Predicate p,
  393. associative_container_tag, unstable_tag)
  394. {
  395. // This method is really slow, so hopefully we won't have any
  396. // associative containers with unstable iterators!
  397. // Is there a better way to do this?
  398. typename AssociativeContainer::iterator i;
  399. typename AssociativeContainer::size_type n = c.size();
  400. while (n--)
  401. for (i = c.begin(); i != c.end(); ++i)
  402. if (p(*i)) {
  403. c.erase(i);
  404. break;
  405. }
  406. }
  407. template <class Container, class Predicate>
  408. void erase_if(Container& c, Predicate p)
  409. {
  410. erase_if_dispatch(c, p, container_category(c), iterator_stability(c));
  411. }
  412. // Push
  413. template <class Container, class T>
  414. std::pair<typename Container::iterator, bool>
  415. push_dispatch(Container& c, BOOST_PENDING_FWD_TYPE(T) v, back_insertion_sequence_tag)
  416. {
  417. c.push_back(BOOST_PENDING_FWD_VALUE(T, v));
  418. return std::make_pair(boost::prior(c.end()), true);
  419. }
  420. template <class Container, class T>
  421. std::pair<typename Container::iterator, bool>
  422. push_dispatch(Container& c, BOOST_PENDING_FWD_TYPE(T) v, front_insertion_sequence_tag)
  423. {
  424. c.push_front(BOOST_PENDING_FWD_VALUE(T, v));
  425. return std::make_pair(c.begin(), true);
  426. }
  427. template <class AssociativeContainer, class T>
  428. std::pair<typename AssociativeContainer::iterator, bool>
  429. push_dispatch(AssociativeContainer& c, BOOST_PENDING_FWD_TYPE(T) v,
  430. unique_associative_container_tag)
  431. {
  432. return c.insert(BOOST_PENDING_FWD_VALUE(T, v));
  433. }
  434. template <class AssociativeContainer, class T>
  435. std::pair<typename AssociativeContainer::iterator, bool>
  436. push_dispatch(AssociativeContainer& c, BOOST_PENDING_FWD_TYPE(T) v,
  437. multiple_associative_container_tag)
  438. {
  439. return std::make_pair(c.insert(BOOST_PENDING_FWD_VALUE(T, v)), true);
  440. }
  441. template <class Container, class T>
  442. std::pair<typename Container::iterator,bool>
  443. push(Container& c, BOOST_PENDING_FWD_TYPE(T) v)
  444. {
  445. return push_dispatch(c, BOOST_PENDING_FWD_VALUE(T, v), container_category(c));
  446. }
  447. // Find
  448. template <class Container, class Value>
  449. typename Container::iterator
  450. find_dispatch(Container& c,
  451. const Value& value,
  452. container_tag)
  453. {
  454. return std::find(c.begin(), c.end(), value);
  455. }
  456. template <class AssociativeContainer, class Value>
  457. typename AssociativeContainer::iterator
  458. find_dispatch(AssociativeContainer& c,
  459. const Value& value,
  460. associative_container_tag)
  461. {
  462. return c.find(value);
  463. }
  464. template <class Container, class Value>
  465. typename Container::iterator
  466. find(Container& c,
  467. const Value& value)
  468. {
  469. return find_dispatch(c, value,
  470. graph_detail::container_category(c));
  471. }
  472. // Find (const versions)
  473. template <class Container, class Value>
  474. typename Container::const_iterator
  475. find_dispatch(const Container& c,
  476. const Value& value,
  477. container_tag)
  478. {
  479. return std::find(c.begin(), c.end(), value);
  480. }
  481. template <class AssociativeContainer, class Value>
  482. typename AssociativeContainer::const_iterator
  483. find_dispatch(const AssociativeContainer& c,
  484. const Value& value,
  485. associative_container_tag)
  486. {
  487. return c.find(value);
  488. }
  489. template <class Container, class Value>
  490. typename Container::const_iterator
  491. find(const Container& c,
  492. const Value& value)
  493. {
  494. return find_dispatch(c, value,
  495. graph_detail::container_category(c));
  496. }
  497. // Equal range
  498. #if 0
  499. // Make the dispatch fail if c is not an Associative Container (and thus
  500. // doesn't have equal_range unless it is sorted, which we cannot check
  501. // statically and is not typically true for BGL's uses of this function).
  502. template <class Container,
  503. class LessThanComparable>
  504. std::pair<typename Container::iterator, typename Container::iterator>
  505. equal_range_dispatch(Container& c,
  506. const LessThanComparable& value,
  507. container_tag)
  508. {
  509. // c must be sorted for std::equal_range to behave properly.
  510. return std::equal_range(c.begin(), c.end(), value);
  511. }
  512. #endif
  513. template <class AssociativeContainer, class Value>
  514. std::pair<typename AssociativeContainer::iterator,
  515. typename AssociativeContainer::iterator>
  516. equal_range_dispatch(AssociativeContainer& c,
  517. const Value& value,
  518. associative_container_tag)
  519. {
  520. return c.equal_range(value);
  521. }
  522. template <class Container, class Value>
  523. std::pair<typename Container::iterator, typename Container::iterator>
  524. equal_range(Container& c,
  525. const Value& value)
  526. {
  527. return equal_range_dispatch(c, value,
  528. graph_detail::container_category(c));
  529. }
  530. }} // namespace boost::graph_detail
  531. #undef BOOST_PENDING_FWD_TYPE
  532. #undef BOOST_PENDING_FWD_VALUE
  533. #endif // BOOST_GRAPH_DETAIL_CONTAINER_TRAITS_H