minmax_element.hpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553
  1. // (C) Copyright Herve Bronnimann 2004.
  2. //
  3. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  4. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  5. /*
  6. Revision history:
  7. 1 July 2004
  8. Split the code into two headers to lessen dependence on
  9. Boost.tuple. (Herve)
  10. 26 June 2004
  11. Added the code for the boost minmax library. (Herve)
  12. */
  13. #ifndef BOOST_ALGORITHM_MINMAX_ELEMENT_HPP
  14. #define BOOST_ALGORITHM_MINMAX_ELEMENT_HPP
  15. /* PROPOSED STANDARD EXTENSIONS:
  16. *
  17. * minmax_element(first, last)
  18. * Effect: std::make_pair( std::min_element(first, last),
  19. * std::max_element(first, last) );
  20. *
  21. * minmax_element(first, last, comp)
  22. * Effect: std::make_pair( std::min_element(first, last, comp),
  23. * std::max_element(first, last, comp) );
  24. */
  25. #include <utility> // for std::pair and std::make_pair
  26. namespace boost {
  27. namespace detail { // for obtaining a uniform version of minmax_element
  28. // that compiles with VC++ 6.0 -- avoid the iterator_traits by
  29. // having comparison object over iterator, not over dereferenced value
  30. template <typename Iterator>
  31. struct less_over_iter {
  32. bool operator()(Iterator const& it1,
  33. Iterator const& it2) const { return *it1 < *it2; }
  34. };
  35. template <typename Iterator, class BinaryPredicate>
  36. struct binary_pred_over_iter {
  37. explicit binary_pred_over_iter(BinaryPredicate const& p ) : m_p( p ) {}
  38. bool operator()(Iterator const& it1,
  39. Iterator const& it2) const { return m_p(*it1, *it2); }
  40. private:
  41. BinaryPredicate m_p;
  42. };
  43. // common base for the two minmax_element overloads
  44. template <typename ForwardIter, class Compare >
  45. std::pair<ForwardIter,ForwardIter>
  46. basic_minmax_element(ForwardIter first, ForwardIter last, Compare comp)
  47. {
  48. if (first == last)
  49. return std::make_pair(last,last);
  50. ForwardIter min_result = first;
  51. ForwardIter max_result = first;
  52. // if only one element
  53. ForwardIter second = first; ++second;
  54. if (second == last)
  55. return std::make_pair(min_result, max_result);
  56. // treat first pair separately (only one comparison for first two elements)
  57. ForwardIter potential_min_result = last;
  58. if (comp(first, second))
  59. max_result = second;
  60. else {
  61. min_result = second;
  62. potential_min_result = first;
  63. }
  64. // then each element by pairs, with at most 3 comparisons per pair
  65. first = ++second; if (first != last) ++second;
  66. while (second != last) {
  67. if (comp(first, second)) {
  68. if (comp(first, min_result)) {
  69. min_result = first;
  70. potential_min_result = last;
  71. }
  72. if (comp(max_result, second))
  73. max_result = second;
  74. } else {
  75. if (comp(second, min_result)) {
  76. min_result = second;
  77. potential_min_result = first;
  78. }
  79. if (comp(max_result, first))
  80. max_result = first;
  81. }
  82. first = ++second;
  83. if (first != last) ++second;
  84. }
  85. // if odd number of elements, treat last element
  86. if (first != last) { // odd number of elements
  87. if (comp(first, min_result)) {
  88. min_result = first;
  89. potential_min_result = last;
  90. }
  91. else if (comp(max_result, first))
  92. max_result = first;
  93. }
  94. // resolve min_result being incorrect with one extra comparison
  95. // (in which case potential_min_result is necessarily the correct result)
  96. if (potential_min_result != last
  97. && !comp(min_result, potential_min_result))
  98. min_result = potential_min_result;
  99. return std::make_pair(min_result,max_result);
  100. }
  101. } // namespace detail
  102. template <typename ForwardIter>
  103. std::pair<ForwardIter,ForwardIter>
  104. minmax_element(ForwardIter first, ForwardIter last)
  105. {
  106. return detail::basic_minmax_element(first, last,
  107. detail::less_over_iter<ForwardIter>() );
  108. }
  109. template <typename ForwardIter, class BinaryPredicate>
  110. std::pair<ForwardIter,ForwardIter>
  111. minmax_element(ForwardIter first, ForwardIter last, BinaryPredicate comp)
  112. {
  113. return detail::basic_minmax_element(first, last,
  114. detail::binary_pred_over_iter<ForwardIter,BinaryPredicate>(comp) );
  115. }
  116. }
  117. /* PROPOSED BOOST EXTENSIONS
  118. * In the description below, [rfirst,rlast) denotes the reversed range
  119. * of [first,last). Even though the iterator type of first and last may
  120. * be only a Forward Iterator, it is possible to explain the semantics
  121. * by assuming that it is a Bidirectional Iterator. In the sequel,
  122. * reverse(ForwardIterator&) returns the reverse_iterator adaptor.
  123. * This is not how the functions would be implemented!
  124. *
  125. * first_min_element(first, last)
  126. * Effect: std::min_element(first, last);
  127. *
  128. * first_min_element(first, last, comp)
  129. * Effect: std::min_element(first, last, comp);
  130. *
  131. * last_min_element(first, last)
  132. * Effect: reverse( std::min_element(reverse(last), reverse(first)) );
  133. *
  134. * last_min_element(first, last, comp)
  135. * Effect: reverse( std::min_element(reverse(last), reverse(first), comp) );
  136. *
  137. * first_max_element(first, last)
  138. * Effect: std::max_element(first, last);
  139. *
  140. * first_max_element(first, last, comp)
  141. * Effect: max_element(first, last);
  142. *
  143. * last_max_element(first, last)
  144. * Effect: reverse( std::max_element(reverse(last), reverse(first)) );
  145. *
  146. * last_max_element(first, last, comp)
  147. * Effect: reverse( std::max_element(reverse(last), reverse(first), comp) );
  148. *
  149. * first_min_first_max_element(first, last)
  150. * Effect: std::make_pair( first_min_element(first, last),
  151. * first_max_element(first, last) );
  152. *
  153. * first_min_first_max_element(first, last, comp)
  154. * Effect: std::make_pair( first_min_element(first, last, comp),
  155. * first_max_element(first, last, comp) );
  156. *
  157. * first_min_last_max_element(first, last)
  158. * Effect: std::make_pair( first_min_element(first, last),
  159. * last_max_element(first, last) );
  160. *
  161. * first_min_last_max_element(first, last, comp)
  162. * Effect: std::make_pair( first_min_element(first, last, comp),
  163. * last_max_element(first, last, comp) );
  164. *
  165. * last_min_first_max_element(first, last)
  166. * Effect: std::make_pair( last_min_element(first, last),
  167. * first_max_element(first, last) );
  168. *
  169. * last_min_first_max_element(first, last, comp)
  170. * Effect: std::make_pair( last_min_element(first, last, comp),
  171. * first_max_element(first, last, comp) );
  172. *
  173. * last_min_last_max_element(first, last)
  174. * Effect: std::make_pair( last_min_element(first, last),
  175. * last_max_element(first, last) );
  176. *
  177. * last_min_last_max_element(first, last, comp)
  178. * Effect: std::make_pair( last_min_element(first, last, comp),
  179. * last_max_element(first, last, comp) );
  180. */
  181. namespace boost {
  182. // Min_element and max_element variants
  183. namespace detail { // common base for the overloads
  184. template <typename ForwardIter, class BinaryPredicate>
  185. ForwardIter
  186. basic_first_min_element(ForwardIter first, ForwardIter last,
  187. BinaryPredicate comp)
  188. {
  189. if (first == last) return last;
  190. ForwardIter min_result = first;
  191. while (++first != last)
  192. if (comp(first, min_result))
  193. min_result = first;
  194. return min_result;
  195. }
  196. template <typename ForwardIter, class BinaryPredicate>
  197. ForwardIter
  198. basic_last_min_element(ForwardIter first, ForwardIter last,
  199. BinaryPredicate comp)
  200. {
  201. if (first == last) return last;
  202. ForwardIter min_result = first;
  203. while (++first != last)
  204. if (!comp(min_result, first))
  205. min_result = first;
  206. return min_result;
  207. }
  208. template <typename ForwardIter, class BinaryPredicate>
  209. ForwardIter
  210. basic_first_max_element(ForwardIter first, ForwardIter last,
  211. BinaryPredicate comp)
  212. {
  213. if (first == last) return last;
  214. ForwardIter max_result = first;
  215. while (++first != last)
  216. if (comp(max_result, first))
  217. max_result = first;
  218. return max_result;
  219. }
  220. template <typename ForwardIter, class BinaryPredicate>
  221. ForwardIter
  222. basic_last_max_element(ForwardIter first, ForwardIter last,
  223. BinaryPredicate comp)
  224. {
  225. if (first == last) return last;
  226. ForwardIter max_result = first;
  227. while (++first != last)
  228. if (!comp(first, max_result))
  229. max_result = first;
  230. return max_result;
  231. }
  232. } // namespace detail
  233. template <typename ForwardIter>
  234. ForwardIter
  235. first_min_element(ForwardIter first, ForwardIter last)
  236. {
  237. return detail::basic_first_min_element(first, last,
  238. detail::less_over_iter<ForwardIter>() );
  239. }
  240. template <typename ForwardIter, class BinaryPredicate>
  241. ForwardIter
  242. first_min_element(ForwardIter first, ForwardIter last, BinaryPredicate comp)
  243. {
  244. return detail::basic_first_min_element(first, last,
  245. detail::binary_pred_over_iter<ForwardIter,BinaryPredicate>(comp) );
  246. }
  247. template <typename ForwardIter>
  248. ForwardIter
  249. last_min_element(ForwardIter first, ForwardIter last)
  250. {
  251. return detail::basic_last_min_element(first, last,
  252. detail::less_over_iter<ForwardIter>() );
  253. }
  254. template <typename ForwardIter, class BinaryPredicate>
  255. ForwardIter
  256. last_min_element(ForwardIter first, ForwardIter last, BinaryPredicate comp)
  257. {
  258. return detail::basic_last_min_element(first, last,
  259. detail::binary_pred_over_iter<ForwardIter,BinaryPredicate>(comp) );
  260. }
  261. template <typename ForwardIter>
  262. ForwardIter
  263. first_max_element(ForwardIter first, ForwardIter last)
  264. {
  265. return detail::basic_first_max_element(first, last,
  266. detail::less_over_iter<ForwardIter>() );
  267. }
  268. template <typename ForwardIter, class BinaryPredicate>
  269. ForwardIter
  270. first_max_element(ForwardIter first, ForwardIter last, BinaryPredicate comp)
  271. {
  272. return detail::basic_first_max_element(first, last,
  273. detail::binary_pred_over_iter<ForwardIter,BinaryPredicate>(comp) );
  274. }
  275. template <typename ForwardIter>
  276. ForwardIter
  277. last_max_element(ForwardIter first, ForwardIter last)
  278. {
  279. return detail::basic_last_max_element(first, last,
  280. detail::less_over_iter<ForwardIter>() );
  281. }
  282. template <typename ForwardIter, class BinaryPredicate>
  283. ForwardIter
  284. last_max_element(ForwardIter first, ForwardIter last, BinaryPredicate comp)
  285. {
  286. return detail::basic_last_max_element(first, last,
  287. detail::binary_pred_over_iter<ForwardIter,BinaryPredicate>(comp) );
  288. }
  289. // Minmax_element variants -- comments removed
  290. namespace detail {
  291. template <typename ForwardIter, class BinaryPredicate>
  292. std::pair<ForwardIter,ForwardIter>
  293. basic_first_min_last_max_element(ForwardIter first, ForwardIter last,
  294. BinaryPredicate comp)
  295. {
  296. if (first == last)
  297. return std::make_pair(last,last);
  298. ForwardIter min_result = first;
  299. ForwardIter max_result = first;
  300. ForwardIter second = ++first;
  301. if (second == last)
  302. return std::make_pair(min_result, max_result);
  303. if (comp(second, min_result))
  304. min_result = second;
  305. else
  306. max_result = second;
  307. first = ++second; if (first != last) ++second;
  308. while (second != last) {
  309. if (!comp(second, first)) {
  310. if (comp(first, min_result))
  311. min_result = first;
  312. if (!comp(second, max_result))
  313. max_result = second;
  314. } else {
  315. if (comp(second, min_result))
  316. min_result = second;
  317. if (!comp(first, max_result))
  318. max_result = first;
  319. }
  320. first = ++second; if (first != last) ++second;
  321. }
  322. if (first != last) {
  323. if (comp(first, min_result))
  324. min_result = first;
  325. else if (!comp(first, max_result))
  326. max_result = first;
  327. }
  328. return std::make_pair(min_result, max_result);
  329. }
  330. template <typename ForwardIter, class BinaryPredicate>
  331. std::pair<ForwardIter,ForwardIter>
  332. basic_last_min_first_max_element(ForwardIter first, ForwardIter last,
  333. BinaryPredicate comp)
  334. {
  335. if (first == last) return std::make_pair(last,last);
  336. ForwardIter min_result = first;
  337. ForwardIter max_result = first;
  338. ForwardIter second = ++first;
  339. if (second == last)
  340. return std::make_pair(min_result, max_result);
  341. if (comp(max_result, second))
  342. max_result = second;
  343. else
  344. min_result = second;
  345. first = ++second; if (first != last) ++second;
  346. while (second != last) {
  347. if (comp(first, second)) {
  348. if (!comp(min_result, first))
  349. min_result = first;
  350. if (comp(max_result, second))
  351. max_result = second;
  352. } else {
  353. if (!comp(min_result, second))
  354. min_result = second;
  355. if (comp(max_result, first))
  356. max_result = first;
  357. }
  358. first = ++second; if (first != last) ++second;
  359. }
  360. if (first != last) {
  361. if (!comp(min_result, first))
  362. min_result = first;
  363. else if (comp(max_result, first))
  364. max_result = first;
  365. }
  366. return std::make_pair(min_result, max_result);
  367. }
  368. template <typename ForwardIter, class BinaryPredicate>
  369. std::pair<ForwardIter,ForwardIter>
  370. basic_last_min_last_max_element(ForwardIter first, ForwardIter last,
  371. BinaryPredicate comp)
  372. {
  373. if (first == last) return std::make_pair(last,last);
  374. ForwardIter min_result = first;
  375. ForwardIter max_result = first;
  376. ForwardIter second = first; ++second;
  377. if (second == last)
  378. return std::make_pair(min_result,max_result);
  379. ForwardIter potential_max_result = last;
  380. if (comp(first, second))
  381. max_result = second;
  382. else {
  383. min_result = second;
  384. potential_max_result = second;
  385. }
  386. first = ++second; if (first != last) ++second;
  387. while (second != last) {
  388. if (comp(first, second)) {
  389. if (!comp(min_result, first))
  390. min_result = first;
  391. if (!comp(second, max_result)) {
  392. max_result = second;
  393. potential_max_result = last;
  394. }
  395. } else {
  396. if (!comp(min_result, second))
  397. min_result = second;
  398. if (!comp(first, max_result)) {
  399. max_result = first;
  400. potential_max_result = second;
  401. }
  402. }
  403. first = ++second;
  404. if (first != last) ++second;
  405. }
  406. if (first != last) {
  407. if (!comp(min_result, first))
  408. min_result = first;
  409. if (!comp(first, max_result)) {
  410. max_result = first;
  411. potential_max_result = last;
  412. }
  413. }
  414. if (potential_max_result != last
  415. && !comp(potential_max_result, max_result))
  416. max_result = potential_max_result;
  417. return std::make_pair(min_result,max_result);
  418. }
  419. } // namespace detail
  420. template <typename ForwardIter>
  421. inline std::pair<ForwardIter,ForwardIter>
  422. first_min_first_max_element(ForwardIter first, ForwardIter last)
  423. {
  424. return minmax_element(first, last);
  425. }
  426. template <typename ForwardIter, class BinaryPredicate>
  427. inline std::pair<ForwardIter,ForwardIter>
  428. first_min_first_max_element(ForwardIter first, ForwardIter last,
  429. BinaryPredicate comp)
  430. {
  431. return minmax_element(first, last, comp);
  432. }
  433. template <typename ForwardIter>
  434. std::pair<ForwardIter,ForwardIter>
  435. first_min_last_max_element(ForwardIter first, ForwardIter last)
  436. {
  437. return detail::basic_first_min_last_max_element(first, last,
  438. detail::less_over_iter<ForwardIter>() );
  439. }
  440. template <typename ForwardIter, class BinaryPredicate>
  441. inline std::pair<ForwardIter,ForwardIter>
  442. first_min_last_max_element(ForwardIter first, ForwardIter last,
  443. BinaryPredicate comp)
  444. {
  445. return detail::basic_first_min_last_max_element(first, last,
  446. detail::binary_pred_over_iter<ForwardIter,BinaryPredicate>(comp) );
  447. }
  448. template <typename ForwardIter>
  449. std::pair<ForwardIter,ForwardIter>
  450. last_min_first_max_element(ForwardIter first, ForwardIter last)
  451. {
  452. return detail::basic_last_min_first_max_element(first, last,
  453. detail::less_over_iter<ForwardIter>() );
  454. }
  455. template <typename ForwardIter, class BinaryPredicate>
  456. inline std::pair<ForwardIter,ForwardIter>
  457. last_min_first_max_element(ForwardIter first, ForwardIter last,
  458. BinaryPredicate comp)
  459. {
  460. return detail::basic_last_min_first_max_element(first, last,
  461. detail::binary_pred_over_iter<ForwardIter,BinaryPredicate>(comp) );
  462. }
  463. template <typename ForwardIter>
  464. std::pair<ForwardIter,ForwardIter>
  465. last_min_last_max_element(ForwardIter first, ForwardIter last)
  466. {
  467. return detail::basic_last_min_last_max_element(first, last,
  468. detail::less_over_iter<ForwardIter>() );
  469. }
  470. template <typename ForwardIter, class BinaryPredicate>
  471. inline std::pair<ForwardIter,ForwardIter>
  472. last_min_last_max_element(ForwardIter first, ForwardIter last,
  473. BinaryPredicate comp)
  474. {
  475. return detail::basic_last_min_last_max_element(first, last,
  476. detail::binary_pred_over_iter<ForwardIter,BinaryPredicate>(comp) );
  477. }
  478. } // namespace boost
  479. #endif // BOOST_ALGORITHM_MINMAX_ELEMENT_HPP