mirror of
https://github.com/boostorg/algorithm.git
synced 2025-06-25 20:11:50 +02:00
Compare commits
110 Commits
indirectSo
...
boost-1.80
Author | SHA1 | Date | |
---|---|---|---|
1c5f74ae25 | |||
1cfd9592f7 | |||
559835eaa5 | |||
3b3bd8d3db | |||
ca77e55428 | |||
7f527f3c4d | |||
e59958cc4d | |||
44527f4c81 | |||
1b65b8de02 | |||
5af3e3b174 | |||
badf1b3f0c | |||
4f9afb0682 | |||
6d0126df8a | |||
daef6d41c5 | |||
055075aa61 | |||
d029402f05 | |||
24d2cceb27 | |||
c5b41eba46 | |||
e5ea93bab1 | |||
5412438df5 | |||
baa6eca18c | |||
073eb62f64 | |||
cc1392cae6 | |||
a09963bf93 | |||
cf249c090c | |||
645be22fa7 | |||
28b12d7264 | |||
4d28d579e3 | |||
5adab54486 | |||
685a76f094 | |||
5988a55b96 | |||
850fc02667 | |||
5279c8f061 | |||
8b89b5ba27 | |||
ca23b6f4f8 | |||
55cb3afefa | |||
beeedadba9 | |||
1a70166889 | |||
63da6f5713 | |||
2381d0bdac | |||
40b5941652 | |||
00dfda98b2 | |||
52eef989da | |||
8132864884 | |||
6e098b27aa | |||
60010b4165 | |||
1660dc9d48 | |||
5ae4f848b3 | |||
fe3e0bb9c4 | |||
311e169376 | |||
3dddfa1930 | |||
be6d8f9665 | |||
bced4ed8dd | |||
1b57e905ab | |||
29bd9f53d9 | |||
6341cfb1a6 | |||
7f4acd6170 | |||
314f6dcfe0 | |||
167aa6e31c | |||
d228e91494 | |||
9cc573fbd0 | |||
28a7d3eb4b | |||
883cce61a8 | |||
96d4708367 | |||
563fe27a59 | |||
76cd99ed53 | |||
0f2399fef0 | |||
044d667e79 | |||
be9da63894 | |||
787c94bc53 | |||
e87ce37b34 | |||
199a89a1e9 | |||
01492a93c6 | |||
50703b8c97 | |||
0f8d556130 | |||
bbd3220a1e | |||
9068069106 | |||
a37af3c81e | |||
f5885c6fb0 | |||
d45bb3545e | |||
d735b9fa1e | |||
62ec675581 | |||
e7cd4da67b | |||
6076f5a18e | |||
60cd5a0500 | |||
c33dad924d | |||
2f2935f07e | |||
3cbaafc27f | |||
c067b348bf | |||
c33935fa1f | |||
98a8b08afb | |||
fc0f3dcffc | |||
822636418b | |||
352e16aade | |||
89c76ea1bb | |||
50b5726a6f | |||
d4b95734dd | |||
05af96f84c | |||
5bdbb2b308 | |||
1a02969303 | |||
6309379618 | |||
37581bac55 | |||
a71a4ed5b1 | |||
c509c3fbad | |||
d8683f2498 | |||
7c0101aa51 | |||
6f3e85528f | |||
8af639b7cf | |||
d9bc7e800b | |||
b4ed9beb90 |
@ -16,18 +16,18 @@ using quickbook ;
|
||||
using doxygen ;
|
||||
using boostbook ;
|
||||
|
||||
doxygen autodoc
|
||||
:
|
||||
[ glob ../../../boost/algorithm/*.hpp
|
||||
doxygen autodoc
|
||||
:
|
||||
[ glob ../../../boost/algorithm/*.hpp
|
||||
../../../boost/algorithm/searching/*.hpp
|
||||
../../../boost/algorithm/cxx11/*.hpp
|
||||
../../../boost/algorithm/cxx14/*.hpp
|
||||
../../../boost/algorithm/cxx17/*.hpp
|
||||
]
|
||||
:
|
||||
:
|
||||
<doxygen:param>"PREDEFINED=\"BOOST_ALGORITHM_DOXYGEN=1\""
|
||||
<doxygen:param>WARNINGS=YES # Default NO, but useful to see warnings, especially in a logfile.
|
||||
;
|
||||
;
|
||||
|
||||
|
||||
xml algorithm : algorithm.qbk ;
|
||||
@ -36,7 +36,7 @@ boostbook standalone
|
||||
:
|
||||
algorithm
|
||||
:
|
||||
<dependency>autodoc
|
||||
<dependency>autodoc
|
||||
<xsl:param>boost.root=../../../..
|
||||
<xsl:param>"boost.doxygen.reftitle=Boost.Algorithms C++ Reference"
|
||||
<xsl:param>chapter.autolabel=0
|
||||
|
@ -233,8 +233,6 @@ Convert a sequence of hexadecimal characters into a sequence of integers or char
|
||||
Convert a sequence of integral types into a lower case hexadecimal sequence of characters
|
||||
[endsect:hex_lower]
|
||||
|
||||
[include indirect_sort.qbk]
|
||||
|
||||
[include is_palindrome.qbk]
|
||||
|
||||
[include is_partitioned_until.qbk]
|
||||
|
@ -16,12 +16,12 @@ The routine `apply_permutation` takes a item sequence and a order sequence. It r
|
||||
The routine `apply_reverse_permutation` takes a item sequence and a order sequence. It will reshuffle item sequence according to order sequence. Every value in order sequence means where the item goes to. Order sequence needs to be exactly a permutation of the sequence [0, 1, ... , N], where N is the biggest index in the item sequence (zero-indexed).
|
||||
|
||||
Implementations are based on these articles:
|
||||
https://devblogs.microsoft.com/oldnewthing/20170102-00/?p=95095
|
||||
https://devblogs.microsoft.com/oldnewthing/20170103-00/?p=95105
|
||||
https://devblogs.microsoft.com/oldnewthing/20170104-00/?p=95115
|
||||
https://devblogs.microsoft.com/oldnewthing/20170109-00/?p=95145
|
||||
https://devblogs.microsoft.com/oldnewthing/20170110-00/?p=95155
|
||||
https://devblogs.microsoft.com/oldnewthing/20170111-00/?p=95165
|
||||
https://blogs.msdn.microsoft.com/oldnewthing/20170102-00/?p=95095
|
||||
https://blogs.msdn.microsoft.com/oldnewthing/20170103-00/?p=95105
|
||||
https://blogs.msdn.microsoft.com/oldnewthing/20170104-00/?p=95115
|
||||
https://blogs.msdn.microsoft.com/oldnewthing/20170109-00/?p=95145
|
||||
https://blogs.msdn.microsoft.com/oldnewthing/20170110-00/?p=95155
|
||||
https://blogs.msdn.microsoft.com/oldnewthing/20170111-00/?p=95165
|
||||
|
||||
The routines come in 2 forms; the first one takes two iterators to define the item range and one iterator to define the beginning of index range. The second form takes range to define the item sequence and range to define index sequence.
|
||||
|
||||
|
@ -1,111 +0,0 @@
|
||||
[/ File indirect_sort.qbk]
|
||||
|
||||
[section:indirect_sort indirect_sort ]
|
||||
|
||||
[/license
|
||||
Copyright (c) 2023 Marshall Clow
|
||||
|
||||
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)
|
||||
]
|
||||
|
||||
There are times that you want a sorted version of a sequence, but for some reason you don't want to modify it. Maybe the elements in the sequence can't be moved/copied, e.g. the sequence is const, or they're just really expensive to move around. An example of this might be a sequence of records from a database.
|
||||
|
||||
That's where indirect sorting comes in. In a "normal" sort, the elements of the sequence to be sorted are shuffled in place. In indirect sorting, the elements are unchanged, but the sort algorithm returns a "permutation" of the elements that, when applied, will put the elements in the sequence in a sorted order.
|
||||
|
||||
Assume have a sequence `[first, last)` of 1000 items that are expensive to swap:
|
||||
```
|
||||
std::sort(first, last); // ['O(N ln N)] comparisons and ['O(N ln N)] swaps (of the element type).
|
||||
```
|
||||
|
||||
On the other hand, using indirect sorting:
|
||||
```
|
||||
auto perm = indirect_sort(first, last); // ['O(N lg N)] comparisons and ['O(N lg N)] swaps (of size_t).
|
||||
apply_permutation(first, last, perm.begin(), perm.end()); // ['O(N)] swaps (of the element type)
|
||||
```
|
||||
|
||||
If the element type is sufficiently expensive to swap, then 10,000 swaps of size_t + 1000 swaps of the element_type could be cheaper than 10,000 swaps of the element_type.
|
||||
|
||||
Or maybe you don't need the elements to actually be sorted - you just want to traverse them in a sorted order:
|
||||
```
|
||||
auto permutation = indirect_sort(first, last);
|
||||
for (size_t idx: permutation)
|
||||
std::cout << first[idx] << std::endl;
|
||||
```
|
||||
|
||||
|
||||
Assume that instead of an "array of structures", you have a "struct of arrays".
|
||||
```
|
||||
struct AType {
|
||||
Type0 key;
|
||||
Type1 value1;
|
||||
Type1 value2;
|
||||
};
|
||||
|
||||
std::array<AType, 1000> arrayOfStruct;
|
||||
```
|
||||
|
||||
versus:
|
||||
|
||||
```
|
||||
template <size_t N>
|
||||
struct AType {
|
||||
std::array<Type0, N> key;
|
||||
std::array<Type1, N> value1;
|
||||
std::array<Type2, N> value2;
|
||||
};
|
||||
|
||||
AType<1000> structOfArrays;
|
||||
```
|
||||
|
||||
Sorting the first one is easy, because each set of fields (`key`, `value1`, `value2`) are part of the same struct. But with indirect sorting, the second one is easy to sort as well - just sort the keys, then apply the permutation to the keys and the values:
|
||||
```
|
||||
auto perm = indirect_sort(std::begin(structOfArrays.key), std::end(structOfArrays.key));
|
||||
apply_permutation(structOfArrays.key.begin(), structOfArrays.key.end(), perm.begin(), perm.end());
|
||||
apply_permutation(structOfArrays.value1.begin(), structOfArrays.value1.end(), perm.begin(), perm.end());
|
||||
apply_permutation(structOfArrays.value2.begin(), structOfArrays.value2.end(), perm.begin(), perm.end());
|
||||
```
|
||||
|
||||
[heading interface]
|
||||
|
||||
The function `indirect_sort` returns a `vector<size_t>` containing the permutation necessary to put the input sequence into a sorted order. One version uses `std::less` to do the comparisons; the other lets the caller pass predicate to do the comparisons.
|
||||
|
||||
There is also a variant called `indirect_stable_sort`; it bears the same relation to `indirect_sort` that `std::stable_sort` does to `std::sort`.
|
||||
|
||||
```
|
||||
template <typename RAIterator>
|
||||
std::vector<size_t> indirect_sort (RAIterator first, RAIterator last);
|
||||
|
||||
template <typename RAIterator, typename BinaryPredicate>
|
||||
std::vector<size_t> indirect_sort (RAIterator first, RAIterator last, BinaryPredicate pred);
|
||||
|
||||
template <typename RAIterator>
|
||||
std::vector<size_t> indirect_stable_sort (RAIterator first, RAIterator last);
|
||||
|
||||
template <typename RAIterator, typename BinaryPredicate>
|
||||
std::vector<size_t> indirect_stable_sort (RAIterator first, RAIterator last, BinaryPredicate pred);
|
||||
```
|
||||
|
||||
[heading Examples]
|
||||
|
||||
[heading Iterator Requirements]
|
||||
|
||||
`indirect_sort` requires random-access iterators.
|
||||
|
||||
[heading Complexity]
|
||||
|
||||
Both of the variants of `indirect_sort` run in ['O(N lg N)] time; they are not more (or less) efficient than `std::sort`. There is an extra layer of indirection on each comparison, but all of the swaps are done on values of type `size_t`
|
||||
|
||||
[heading Exception Safety]
|
||||
|
||||
[heading Notes]
|
||||
|
||||
In numpy, this algorithm is known as `argsort`.
|
||||
|
||||
[endsect]
|
||||
|
||||
[/ File indirect_sort.qbk
|
||||
Copyright 2023 Marshall Clow
|
||||
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).
|
||||
]
|
@ -20,7 +20,7 @@
|
||||
#include <functional> // for plus and multiplies
|
||||
|
||||
#include <boost/config.hpp>
|
||||
#include <boost/core/enable_if.hpp> // for boost::disable_if
|
||||
#include <boost/utility/enable_if.hpp> // for boost::disable_if
|
||||
#include <boost/type_traits/is_integral.hpp>
|
||||
|
||||
namespace boost { namespace algorithm {
|
||||
|
@ -27,7 +27,7 @@
|
||||
#include <boost/range/begin.hpp>
|
||||
#include <boost/range/end.hpp>
|
||||
#include <boost/type_traits/type_identity.hpp> // for boost::type_identity
|
||||
#include <boost/core/enable_if.hpp> // for boost::disable_if
|
||||
#include <boost/utility/enable_if.hpp> // for boost::disable_if
|
||||
|
||||
namespace boost { namespace algorithm {
|
||||
|
||||
|
@ -20,7 +20,7 @@
|
||||
#include <boost/config.hpp>
|
||||
#include <boost/range/begin.hpp>
|
||||
#include <boost/range/end.hpp>
|
||||
#include <boost/core/enable_if.hpp>
|
||||
#include <boost/utility/enable_if.hpp>
|
||||
#include <boost/type_traits/is_same.hpp>
|
||||
|
||||
namespace boost { namespace algorithm {
|
||||
|
@ -20,7 +20,7 @@
|
||||
#include <boost/range/begin.hpp>
|
||||
#include <boost/range/end.hpp>
|
||||
|
||||
#include <boost/core/enable_if.hpp>
|
||||
#include <boost/utility/enable_if.hpp>
|
||||
#include <boost/type_traits/is_same.hpp>
|
||||
#include <boost/type_traits/type_identity.hpp> // for boost::type_identity
|
||||
|
||||
|
@ -1,4 +1,4 @@
|
||||
/*
|
||||
/*
|
||||
Copyright 2008 Adobe Systems Incorporated
|
||||
|
||||
Distributed under the Boost Software License, Version 1.0. (See accompanying
|
||||
@ -84,7 +84,7 @@ namespace boost { namespace algorithm {
|
||||
template <
|
||||
typename BidirectionalIterator, // models BidirectionalIterator
|
||||
typename Pred> // models UnaryPredicate
|
||||
std::pair<BidirectionalIterator, BidirectionalIterator> gather
|
||||
std::pair<BidirectionalIterator, BidirectionalIterator> gather
|
||||
( BidirectionalIterator first, BidirectionalIterator last, BidirectionalIterator pivot, Pred pred )
|
||||
{
|
||||
// The first call partitions everything up to (but not including) the pivot element,
|
||||
@ -106,11 +106,11 @@ template <
|
||||
typename BidirectionalRange, //
|
||||
typename Pred> // Pred models UnaryPredicate
|
||||
std::pair<
|
||||
typename boost::range_iterator<BidirectionalRange>::type,
|
||||
typename boost::range_iterator<BidirectionalRange>::type>
|
||||
typename boost::range_iterator<const BidirectionalRange>::type,
|
||||
typename boost::range_iterator<const BidirectionalRange>::type>
|
||||
gather (
|
||||
BidirectionalRange &range,
|
||||
typename boost::range_iterator<BidirectionalRange>::type pivot,
|
||||
const BidirectionalRange &range,
|
||||
typename boost::range_iterator<const BidirectionalRange>::type pivot,
|
||||
Pred pred )
|
||||
{
|
||||
return boost::algorithm::gather ( boost::begin ( range ), boost::end ( range ), pivot, pred );
|
||||
|
@ -30,7 +30,7 @@
|
||||
#include <boost/exception/info.hpp>
|
||||
#include <boost/throw_exception.hpp>
|
||||
|
||||
#include <boost/core/enable_if.hpp>
|
||||
#include <boost/utility/enable_if.hpp>
|
||||
#include <boost/type_traits/is_integral.hpp>
|
||||
|
||||
|
||||
|
@ -1,207 +0,0 @@
|
||||
/*
|
||||
Copyright (c) Marshall Clow 2023.
|
||||
|
||||
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)
|
||||
|
||||
*/
|
||||
|
||||
/// \file indirect_sort.hpp
|
||||
/// \brief indirect sorting algorithms
|
||||
/// \author Marshall Clow
|
||||
///
|
||||
|
||||
#ifndef BOOST_ALGORITHM_INDIRECT_SORT
|
||||
#define BOOST_ALGORITHM_INDIRECT_SORT
|
||||
|
||||
#include <algorithm> // for std::sort (and others)
|
||||
#include <functional> // for std::less
|
||||
#include <vector> // for std::vector
|
||||
|
||||
#include <boost/algorithm/cxx11/iota.hpp>
|
||||
|
||||
namespace boost { namespace algorithm {
|
||||
|
||||
typedef std::vector<size_t> Permutation;
|
||||
|
||||
namespace detail {
|
||||
|
||||
template <class Predicate, class Iter>
|
||||
struct indirect_predicate {
|
||||
indirect_predicate (Predicate pred, Iter iter)
|
||||
: pred_(pred), iter_(iter) {}
|
||||
|
||||
bool operator ()(size_t a, size_t b) const {
|
||||
return pred_(iter_[a], iter_[b]);
|
||||
}
|
||||
|
||||
Predicate pred_;
|
||||
Iter iter_;
|
||||
};
|
||||
|
||||
// Initialize a permutation of size 'size'. [ 0, 1, 2, ... size-1 ]
|
||||
// Note: it would be nice to use 'iota' here, but that call writes over
|
||||
// existing elements - not append them. I don't want to initialize
|
||||
// the elements of the permutation to zero, and then immediately
|
||||
// overwrite them.
|
||||
void init_permutation (Permutation &p, size_t size) {
|
||||
p.reserve(size);
|
||||
boost::algorithm::iota_n(
|
||||
std::back_insert_iterator<Permutation>(p), size_t(0), size);
|
||||
}
|
||||
}
|
||||
|
||||
// ===== sort =====
|
||||
|
||||
/// \fn indirect_sort (RAIterator first, RAIterator last, Predicate pred)
|
||||
/// \returns a permutation of the elements in the range [first, last)
|
||||
/// such that when the permutation is applied to the sequence,
|
||||
/// the result is ordered as if 'std::sort(first, last, pred)'
|
||||
// was called on the sequence.
|
||||
///
|
||||
/// \param first The start of the input sequence
|
||||
/// \param last The end of the input sequence
|
||||
/// \param pred The predicate to compare elements with
|
||||
///
|
||||
template <typename RAIterator, typename Pred>
|
||||
Permutation indirect_sort (RAIterator first, RAIterator last, Pred pred) {
|
||||
|
||||
Permutation ret;
|
||||
detail::init_permutation(ret, std::distance(first, last));
|
||||
std::sort(ret.begin(), ret.end(),
|
||||
detail::indirect_predicate<Pred, RAIterator>(pred, first));
|
||||
return ret;
|
||||
}
|
||||
|
||||
/// \fn indirect_sort (RAIterator first, RAIterator last)
|
||||
/// \returns a permutation of the elements in the range [first, last)
|
||||
/// such that when the permutation is applied to the sequence,
|
||||
/// the result is ordered as if 'std::sort(first, last)'
|
||||
// was called on the sequence.
|
||||
///
|
||||
/// \param first The start of the input sequence
|
||||
/// \param last The end of the input sequence
|
||||
///
|
||||
template <typename RAIterator>
|
||||
Permutation indirect_sort (RAIterator first, RAIterator last) {
|
||||
return indirect_sort(first, last,
|
||||
std::less<typename std::iterator_traits<RAIterator>::value_type>());
|
||||
}
|
||||
|
||||
// ===== stable_sort =====
|
||||
|
||||
/// \fn indirect_stable_sort (RAIterator first, RAIterator last, Predicate pred)
|
||||
/// \returns a permutation of the elements in the range [first, last)
|
||||
/// such that when the permutation is applied to the sequence,
|
||||
/// the result is ordered as if 'std::stable_sort(first, last, pred)'
|
||||
// was called on the sequence.
|
||||
///
|
||||
/// \param first The start of the input sequence
|
||||
/// \param last The end of the input sequence
|
||||
/// \param pred The predicate to compare elements with
|
||||
///
|
||||
template <typename RAIterator, typename Pred>
|
||||
Permutation indirect_stable_sort (RAIterator first, RAIterator last, Pred pred) {
|
||||
Permutation ret;
|
||||
detail::init_permutation(ret, std::distance(first, last));
|
||||
std::stable_sort(ret.begin(), ret.end(),
|
||||
detail::indirect_predicate<Pred, RAIterator>(pred, first));
|
||||
return ret;
|
||||
}
|
||||
|
||||
/// \fn indirect_stable_sort (RAIterator first, RAIterator last)
|
||||
/// \returns a permutation of the elements in the range [first, last)
|
||||
/// such that when the permutation is applied to the sequence,
|
||||
/// the result is ordered as if 'std::stable_sort(first, last)'
|
||||
// was called on the sequence.
|
||||
///
|
||||
/// \param first The start of the input sequence
|
||||
/// \param last The end of the input sequence
|
||||
///
|
||||
template <typename RAIterator>
|
||||
Permutation indirect_stable_sort (RAIterator first, RAIterator last) {
|
||||
return indirect_stable_sort(first, last,
|
||||
std::less<typename std::iterator_traits<RAIterator>::value_type>());
|
||||
}
|
||||
|
||||
// ===== partial_sort =====
|
||||
|
||||
/// \fn indirect_partial_sort (RAIterator first, RAIterator middle, RAIterator last, Predicate pred)
|
||||
/// \returns a permutation of the elements in the range [first, last)
|
||||
/// such that when the permutation is applied to the sequence,
|
||||
/// the result is ordered as if 'std::partial_sort(first, middle, last, pred)'
|
||||
// was called on the sequence.
|
||||
///
|
||||
/// \param first The start of the input sequence
|
||||
/// \param middle The end of the range to be sorted
|
||||
/// \param last The end of the input sequence
|
||||
/// \param pred The predicate to compare elements with
|
||||
///
|
||||
template <typename RAIterator, typename Pred>
|
||||
Permutation indirect_partial_sort (RAIterator first, RAIterator middle,
|
||||
RAIterator last, Pred pred) {
|
||||
Permutation ret;
|
||||
detail::init_permutation(ret, std::distance(first, last));
|
||||
std::partial_sort(ret.begin(), ret.begin() + std::distance(first, middle), ret.end(),
|
||||
detail::indirect_predicate<Pred, RAIterator>(pred, first));
|
||||
return ret;
|
||||
}
|
||||
|
||||
/// \fn indirect_partial_sort (RAIterator first, RAIterator middle, RAIterator last)
|
||||
/// \returns a permutation of the elements in the range [first, last)
|
||||
/// such that when the permutation is applied to the sequence,
|
||||
/// the result is ordered as if 'std::partial_sort(first, middle, last)'
|
||||
// was called on the sequence.
|
||||
///
|
||||
/// \param first The start of the input sequence
|
||||
/// \param middle The end of the range to be sorted
|
||||
/// \param last The end of the input sequence
|
||||
///
|
||||
template <typename RAIterator>
|
||||
Permutation indirect_partial_sort (RAIterator first, RAIterator middle, RAIterator last) {
|
||||
return indirect_partial_sort(first, middle, last,
|
||||
std::less<typename std::iterator_traits<RAIterator>::value_type>());
|
||||
}
|
||||
|
||||
// ===== nth_element =====
|
||||
|
||||
/// \fn indirect_nth_element (RAIterator first, RAIterator nth, RAIterator last, Predicate p)
|
||||
/// \returns a permutation of the elements in the range [first, last)
|
||||
/// such that when the permutation is applied to the sequence,
|
||||
/// the result is ordered as if 'std::nth_element(first, nth, last, p)'
|
||||
// was called on the sequence.
|
||||
///
|
||||
/// \param first The start of the input sequence
|
||||
/// \param nth The sort partition point in the input sequence
|
||||
/// \param last The end of the input sequence
|
||||
/// \param pred The predicate to compare elements with
|
||||
///
|
||||
template <typename RAIterator, typename Pred>
|
||||
Permutation indirect_nth_element (RAIterator first, RAIterator nth,
|
||||
RAIterator last, Pred pred) {
|
||||
Permutation ret;
|
||||
detail::init_permutation(ret, std::distance(first, last));
|
||||
std::nth_element(ret.begin(), ret.begin() + std::distance(first, nth), ret.end(),
|
||||
detail::indirect_predicate<Pred, RAIterator>(pred, first));
|
||||
return ret;
|
||||
}
|
||||
|
||||
/// \fn indirect_nth_element (RAIterator first, RAIterator nth, RAIterator last)
|
||||
/// \returns a permutation of the elements in the range [first, last)
|
||||
/// such that when the permutation is applied to the sequence,
|
||||
/// the result is ordered as if 'std::nth_element(first, nth, last)'
|
||||
// was called on the sequence.
|
||||
///
|
||||
/// \param first The start of the input sequence
|
||||
/// \param nth The sort partition point in the input sequence
|
||||
/// \param last The end of the input sequence
|
||||
///
|
||||
template <typename RAIterator>
|
||||
Permutation indirect_nth_element (RAIterator first, RAIterator nth, RAIterator last) {
|
||||
return indirect_nth_element(first, nth, last,
|
||||
std::less<typename std::iterator_traits<RAIterator>::value_type>());
|
||||
}
|
||||
|
||||
}}
|
||||
|
||||
#endif // BOOST_ALGORITHM_INDIRECT_SORT
|
@ -113,7 +113,7 @@ bool is_palindrome(const R& range, Predicate p)
|
||||
/// \note This function will return true for empty sequences and for palindromes.
|
||||
/// For other sequences function will return false.
|
||||
/// Complexity: O(N).
|
||||
inline bool is_palindrome(const char* str)
|
||||
bool is_palindrome(const char* str)
|
||||
{
|
||||
if(!str)
|
||||
return true;
|
||||
|
@ -19,7 +19,7 @@
|
||||
#include <boost/range/begin.hpp>
|
||||
#include <boost/range/end.hpp>
|
||||
|
||||
#include <boost/core/enable_if.hpp>
|
||||
#include <boost/utility/enable_if.hpp>
|
||||
#include <boost/type_traits/is_same.hpp>
|
||||
|
||||
#include <boost/algorithm/searching/detail/bm_traits.hpp>
|
||||
|
@ -19,7 +19,7 @@
|
||||
#include <boost/range/begin.hpp>
|
||||
#include <boost/range/end.hpp>
|
||||
|
||||
#include <boost/core/enable_if.hpp>
|
||||
#include <boost/utility/enable_if.hpp>
|
||||
#include <boost/type_traits/is_same.hpp>
|
||||
|
||||
#include <boost/algorithm/searching/detail/bm_traits.hpp>
|
||||
|
@ -20,7 +20,7 @@
|
||||
#include <boost/range/begin.hpp>
|
||||
#include <boost/range/end.hpp>
|
||||
|
||||
#include <boost/core/enable_if.hpp>
|
||||
#include <boost/utility/enable_if.hpp>
|
||||
#include <boost/type_traits/is_same.hpp>
|
||||
|
||||
#include <boost/algorithm/searching/detail/debugging.hpp>
|
||||
|
@ -88,10 +88,6 @@ alias unit_test_framework
|
||||
|
||||
# Apply_permutation tests
|
||||
[ run apply_permutation_test.cpp unit_test_framework : : : : apply_permutation_test ]
|
||||
|
||||
# Indirect_sort tests
|
||||
[ run indirect_sort_test.cpp unit_test_framework : : : : indirect_sort_test ]
|
||||
|
||||
# Find tests
|
||||
[ run find_not_test.cpp unit_test_framework : : : : find_not_test ]
|
||||
[ run find_backward_test.cpp unit_test_framework : : : : find_backward_test ]
|
||||
|
@ -28,8 +28,8 @@
|
||||
namespace ba = boost::algorithm;
|
||||
// namespace ba = boost;
|
||||
|
||||
BOOST_CXX14_CONSTEXPR bool is_true ( int ) { return true; }
|
||||
BOOST_CXX14_CONSTEXPR bool is_false ( int ) { return false; }
|
||||
BOOST_CXX14_CONSTEXPR bool is_true ( int v ) { return true; }
|
||||
BOOST_CXX14_CONSTEXPR bool is_false ( int v ) { return false; }
|
||||
BOOST_CXX14_CONSTEXPR bool is_even ( int v ) { return v % 2 == 0; }
|
||||
BOOST_CXX14_CONSTEXPR bool is_odd ( int v ) { return v % 2 == 1; }
|
||||
BOOST_CXX14_CONSTEXPR bool is_zero ( int v ) { return v == 0; }
|
||||
|
@ -25,7 +25,7 @@ void test_short_input1 () {
|
||||
std::string s;
|
||||
|
||||
try { ba::unhex ( std::string ( "A" ), std::back_inserter(s)); }
|
||||
catch ( const std::exception & ) { return; }
|
||||
catch ( const std::exception &ex ) { return; }
|
||||
BOOST_TEST_MESSAGE ( "Failed to catch std::exception in test_short_input1" );
|
||||
BOOST_CHECK ( false );
|
||||
}
|
||||
@ -34,7 +34,7 @@ void test_short_input2 () {
|
||||
std::string s;
|
||||
|
||||
try { ba::unhex ( std::string ( "A" ), std::back_inserter(s)); }
|
||||
catch ( const ba::hex_decode_error & ) { return; }
|
||||
catch ( const ba::hex_decode_error &ex ) { return; }
|
||||
BOOST_TEST_MESSAGE ( "Failed to catch ba::hex_decode_error in test_short_input2" );
|
||||
BOOST_CHECK ( false );
|
||||
}
|
||||
@ -43,7 +43,7 @@ void test_short_input3 () {
|
||||
std::string s;
|
||||
|
||||
try { ba::unhex ( std::string ( "A" ), std::back_inserter(s)); }
|
||||
catch ( const ba::not_enough_input & ) { return; }
|
||||
catch ( const ba::not_enough_input &ex ) { return; }
|
||||
BOOST_TEST_MESSAGE ( "Failed to catch ba::not_enough_input in test_short_input3" );
|
||||
BOOST_CHECK ( false );
|
||||
}
|
||||
@ -53,8 +53,8 @@ void test_short_input4 () {
|
||||
std::string s;
|
||||
|
||||
try { ba::unhex ( std::string ( "A" ), std::back_inserter(s)); }
|
||||
catch ( const ba::non_hex_input & ) { BOOST_CHECK ( false ); }
|
||||
catch ( const ba::not_enough_input & ) { return; }
|
||||
catch ( const ba::non_hex_input &ex ) { BOOST_CHECK ( false ); }
|
||||
catch ( const ba::not_enough_input &ex ) { return; }
|
||||
catch ( ... ) { BOOST_CHECK ( false ); }
|
||||
BOOST_CHECK ( false );
|
||||
}
|
||||
@ -64,8 +64,8 @@ void test_short_input5 () {
|
||||
std::string s;
|
||||
|
||||
try { ba::unhex ( "A", std::back_inserter(s)); }
|
||||
catch ( const ba::non_hex_input & ) { BOOST_CHECK ( false ); }
|
||||
catch ( const ba::not_enough_input & ) { return; }
|
||||
catch ( const ba::non_hex_input &ex ) { BOOST_CHECK ( false ); }
|
||||
catch ( const ba::not_enough_input &ex ) { return; }
|
||||
catch ( ... ) { BOOST_CHECK ( false ); }
|
||||
BOOST_CHECK ( false );
|
||||
}
|
||||
@ -125,8 +125,8 @@ void test_nonhex_input4 () {
|
||||
std::string s;
|
||||
|
||||
try { ba::unhex ( "P1234FA1234", std::back_inserter(s)); }
|
||||
catch ( const ba::not_enough_input & ) { BOOST_CHECK ( false ); }
|
||||
catch ( const ba::non_hex_input & ) { return; }
|
||||
catch ( const ba::not_enough_input &ex ) { BOOST_CHECK ( false ); }
|
||||
catch ( const ba::non_hex_input &ex ) { return; }
|
||||
catch ( ... ) { BOOST_CHECK ( false ); }
|
||||
BOOST_CHECK ( false );
|
||||
}
|
||||
|
@ -1,348 +0,0 @@
|
||||
/*
|
||||
Copyright (c) Marshall Clow 2023.
|
||||
|
||||
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)
|
||||
|
||||
For more information, see http://www.boost.org
|
||||
*/
|
||||
|
||||
#include <boost/config.hpp>
|
||||
#include <boost/algorithm/indirect_sort.hpp>
|
||||
#include <boost/algorithm/apply_permutation.hpp>
|
||||
#include <boost/algorithm/cxx11/is_sorted.hpp>
|
||||
#include <boost/algorithm/cxx11/all_of.hpp>
|
||||
|
||||
#define BOOST_TEST_MAIN
|
||||
#include <boost/test/unit_test.hpp>
|
||||
|
||||
#include <iostream>
|
||||
#include <string>
|
||||
#include <vector>
|
||||
#include <list>
|
||||
|
||||
using boost::algorithm::Permutation;
|
||||
|
||||
// A permutation of size N is a sequence of values in the range [0..N)
|
||||
// such that no value appears more than once in the permutation.
|
||||
bool is_a_permutation(Permutation p, size_t N) {
|
||||
if (p.size() != N) return false;
|
||||
|
||||
// Sort the permutation, and ensure that each value appears exactly once.
|
||||
std::sort(p.begin(), p.end());
|
||||
for (size_t i = 0; i < N; ++i)
|
||||
if (p[i] != i) return false;
|
||||
return true;
|
||||
}
|
||||
|
||||
template <typename Iter,
|
||||
typename Comp = typename std::less<typename std::iterator_traits<Iter>::value_type> >
|
||||
struct indirect_comp {
|
||||
indirect_comp (Iter it, Comp c = Comp())
|
||||
: iter_(it), comp_(c) {}
|
||||
|
||||
bool operator ()(size_t a, size_t b) const { return comp_(iter_[a], iter_[b]);}
|
||||
|
||||
Iter iter_;
|
||||
Comp comp_;
|
||||
};
|
||||
|
||||
//// =======================
|
||||
//// ==== indirect_sort ====
|
||||
//// =======================
|
||||
template <typename Iter>
|
||||
void test_one_sort(Iter first, Iter last) {
|
||||
Permutation perm = boost::algorithm::indirect_sort(first, last);
|
||||
BOOST_CHECK (is_a_permutation(perm, std::distance(first, last)));
|
||||
BOOST_CHECK (boost::algorithm::is_sorted(perm.begin(), perm.end(), indirect_comp<Iter>(first)));
|
||||
|
||||
// Make a copy of the data, apply the permutation, and ensure that it is sorted.
|
||||
typedef std::vector<typename std::iterator_traits<Iter>::value_type> Vector;
|
||||
Vector v(first, last);
|
||||
boost::algorithm::apply_permutation(v.begin(), v.end(), perm.begin(), perm.end());
|
||||
BOOST_CHECK (boost::algorithm::is_sorted(v.begin(), v.end()));
|
||||
}
|
||||
|
||||
template <typename Iter, typename Comp>
|
||||
void test_one_sort(Iter first, Iter last, Comp comp) {
|
||||
Permutation perm = boost::algorithm::indirect_sort(first, last, comp);
|
||||
BOOST_CHECK (is_a_permutation(perm, std::distance(first, last)));
|
||||
BOOST_CHECK (boost::algorithm::is_sorted(perm.begin(), perm.end(),
|
||||
indirect_comp<Iter, Comp>(first, comp)));
|
||||
|
||||
// Make a copy of the data, apply the permutation, and ensure that it is sorted.
|
||||
typedef std::vector<typename std::iterator_traits<Iter>::value_type> Vector;
|
||||
Vector v(first, last);
|
||||
boost::algorithm::apply_permutation(v.begin(), v.end(), perm.begin(), perm.end());
|
||||
BOOST_CHECK (boost::algorithm::is_sorted(v.begin(), v.end(), comp));
|
||||
}
|
||||
|
||||
|
||||
BOOST_AUTO_TEST_CASE(test_sort) {
|
||||
int num[] = { 1,3,5,7,9, 2, 4, 6, 8, 10 };
|
||||
const int sz = sizeof (num)/sizeof(num[0]);
|
||||
int *first = &num[0];
|
||||
int const *cFirst = &num[0];
|
||||
|
||||
// Test subsets
|
||||
for (size_t i = 0; i <= sz; ++i) {
|
||||
test_one_sort(first, first + i);
|
||||
test_one_sort(first, first + i, std::greater<int>());
|
||||
|
||||
// test with constant inputs
|
||||
test_one_sort(cFirst, cFirst + i);
|
||||
test_one_sort(cFirst, cFirst + i, std::greater<int>());
|
||||
}
|
||||
|
||||
// make sure we work with iterators as well as pointers
|
||||
std::vector<int> v(first, first + sz);
|
||||
test_one_sort(v.begin(), v.end());
|
||||
test_one_sort(v.begin(), v.end(), std::greater<int>());
|
||||
}
|
||||
|
||||
|
||||
//// ==============================
|
||||
//// ==== indirect_stable_sort ====
|
||||
//// ==============================
|
||||
|
||||
template <typename T1, typename T2>
|
||||
struct MyPair {
|
||||
MyPair () {}
|
||||
|
||||
MyPair (const T1 &t1, const T2 &t2)
|
||||
: first(t1), second(t2) {}
|
||||
|
||||
T1 first;
|
||||
T2 second;
|
||||
};
|
||||
|
||||
template <typename T1, typename T2>
|
||||
bool operator < (const MyPair<T1, T2>& lhs, const MyPair<T1, T2>& rhs) {
|
||||
return lhs.first < rhs.first; // compare only the first elements
|
||||
}
|
||||
|
||||
template <typename T1, typename T2>
|
||||
bool MyGreater (const MyPair<T1, T2>& lhs, const MyPair<T1, T2>& rhs) {
|
||||
return lhs.first > rhs.first; // compare only the first elements
|
||||
}
|
||||
|
||||
template <typename Iter>
|
||||
void test_one_stable_sort(Iter first, Iter last) {
|
||||
Permutation perm = boost::algorithm::indirect_stable_sort(first, last);
|
||||
BOOST_CHECK (is_a_permutation(perm, std::distance(first, last)));
|
||||
BOOST_CHECK (boost::algorithm::is_sorted(perm.begin(), perm.end(), indirect_comp<Iter>(first)));
|
||||
|
||||
if (first != last) {
|
||||
Iter iFirst = first;
|
||||
Iter iSecond = first; ++iSecond;
|
||||
|
||||
while (iSecond != last) {
|
||||
if (iFirst->first == iSecond->first)
|
||||
BOOST_CHECK(iFirst->second < iSecond->second);
|
||||
++iFirst;
|
||||
++iSecond;
|
||||
}
|
||||
}
|
||||
|
||||
// Make a copy of the data, apply the permutation, and ensure that it is sorted.
|
||||
typedef std::vector<typename std::iterator_traits<Iter>::value_type> Vector;
|
||||
Vector v(first, last);
|
||||
boost::algorithm::apply_permutation(v.begin(), v.end(), perm.begin(), perm.end());
|
||||
BOOST_CHECK (boost::algorithm::is_sorted(v.begin(), v.end()));
|
||||
}
|
||||
|
||||
template <typename Iter, typename Comp>
|
||||
void test_one_stable_sort(Iter first, Iter last, Comp comp) {
|
||||
Permutation perm = boost::algorithm::indirect_stable_sort(first, last, comp);
|
||||
BOOST_CHECK (is_a_permutation(perm, std::distance(first, last)));
|
||||
BOOST_CHECK (boost::algorithm::is_sorted(perm.begin(), perm.end(), indirect_comp<Iter, Comp>(first, comp)));
|
||||
|
||||
if (first != last) {
|
||||
Iter iFirst = first;
|
||||
Iter iSecond = first; ++iSecond;
|
||||
|
||||
while (iSecond != last) {
|
||||
if (iFirst->first == iSecond->first)
|
||||
BOOST_CHECK(iFirst->second < iSecond->second);
|
||||
++iFirst;
|
||||
++iSecond;
|
||||
}
|
||||
}
|
||||
|
||||
// Make a copy of the data, apply the permutation, and ensure that it is sorted.
|
||||
typedef std::vector<typename std::iterator_traits<Iter>::value_type> Vector;
|
||||
Vector v(first, last);
|
||||
boost::algorithm::apply_permutation(v.begin(), v.end(), perm.begin(), perm.end());
|
||||
BOOST_CHECK (boost::algorithm::is_sorted(v.begin(), v.end(), comp));
|
||||
}
|
||||
|
||||
BOOST_AUTO_TEST_CASE(test_stable_sort) {
|
||||
typedef MyPair<int, long> Pair;
|
||||
const int sz = 10;
|
||||
Pair vals[sz];
|
||||
|
||||
for (int i = 0; i < sz; ++i) {
|
||||
vals[i].first = 100 - (i >> 1);
|
||||
vals[i].second = i;
|
||||
}
|
||||
|
||||
Pair *first = &vals[0];
|
||||
Pair const *cFirst = &vals[0];
|
||||
|
||||
// Test subsets
|
||||
for (size_t i = 0; i <= sz; ++i) {
|
||||
test_one_stable_sort(first, first + i);
|
||||
test_one_stable_sort(first, first + i, MyGreater<int, long>);
|
||||
|
||||
// test with constant inputs
|
||||
test_one_sort(cFirst, cFirst + i);
|
||||
test_one_sort(cFirst, cFirst + i, MyGreater<int, long>);
|
||||
}
|
||||
}
|
||||
|
||||
//// ===============================
|
||||
//// ==== indirect_partial_sort ====
|
||||
//// ===============================
|
||||
|
||||
template <typename Iter>
|
||||
void test_one_partial_sort(Iter first, Iter middle, Iter last) {
|
||||
const size_t middleIdx = std::distance(first, middle);
|
||||
Permutation perm = boost::algorithm::indirect_partial_sort(first, middle, last);
|
||||
BOOST_CHECK (is_a_permutation(perm, std::distance(first, last)));
|
||||
BOOST_CHECK (boost::algorithm::is_sorted(perm.begin(), perm.begin() + middleIdx, indirect_comp<Iter>(first)));
|
||||
|
||||
// Make a copy of the data, apply the permutation, and ensure that it is sorted.
|
||||
typedef std::vector<typename std::iterator_traits<Iter>::value_type> Vector;
|
||||
Vector v(first, last);
|
||||
boost::algorithm::apply_permutation(v.begin(), v.end(), perm.begin(), perm.end());
|
||||
BOOST_CHECK (boost::algorithm::is_sorted(v.begin(), v.begin() + middleIdx));
|
||||
|
||||
// Make sure that [middle, end) are all "greater" than the sorted part
|
||||
if (middleIdx > 0) {
|
||||
typename Vector::iterator lastSorted = v.begin() + middleIdx - 1;
|
||||
for (typename Vector::iterator it = v.begin () + middleIdx; it != v.end(); ++it)
|
||||
BOOST_CHECK(*lastSorted < *it);
|
||||
}
|
||||
}
|
||||
|
||||
template <typename Iter, typename Comp>
|
||||
void test_one_partial_sort(Iter first, Iter middle, Iter last, Comp comp) {
|
||||
const size_t middleIdx = std::distance(first, middle);
|
||||
Permutation perm = boost::algorithm::indirect_partial_sort(first, middle, last, comp);
|
||||
BOOST_CHECK (is_a_permutation(perm, std::distance(first, last)));
|
||||
BOOST_CHECK (boost::algorithm::is_sorted(perm.begin(), perm.begin() + middleIdx,
|
||||
indirect_comp<Iter, Comp>(first, comp)));
|
||||
|
||||
// Make a copy of the data, apply the permutation, and ensure that it is sorted.
|
||||
typedef std::vector<typename std::iterator_traits<Iter>::value_type> Vector;
|
||||
Vector v(first, last);
|
||||
boost::algorithm::apply_permutation(v.begin(), v.end(), perm.begin(), perm.end());
|
||||
BOOST_CHECK (boost::algorithm::is_sorted(v.begin(), v.begin() + middleIdx, comp));
|
||||
|
||||
// Make sure that [middle, end) are all "greater" than the sorted part
|
||||
if (middleIdx > 0) {
|
||||
typename Vector::iterator lastSorted = v.begin() + middleIdx - 1;
|
||||
for (typename Vector::iterator it = v.begin () + middleIdx; it != v.end(); ++it)
|
||||
BOOST_CHECK(comp(*lastSorted, *it));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
BOOST_AUTO_TEST_CASE(test_partial_sort) {
|
||||
int num[] = { 1,3,5,7,9, 2, 4, 6, 8, 10 };
|
||||
const int sz = sizeof (num)/sizeof(num[0]);
|
||||
int *first = &num[0];
|
||||
int const *cFirst = &num[0];
|
||||
|
||||
// Test subsets
|
||||
for (size_t i = 0; i <= sz; ++i) {
|
||||
for (size_t j = 0; j < i; ++j) {
|
||||
test_one_partial_sort(first, first + j, first + i);
|
||||
test_one_partial_sort(first, first + j, first + i, std::greater<int>());
|
||||
|
||||
// test with constant inputs
|
||||
test_one_partial_sort(cFirst, cFirst + j, cFirst + i);
|
||||
test_one_partial_sort(cFirst, cFirst + j, cFirst + i, std::greater<int>());
|
||||
}
|
||||
}
|
||||
|
||||
// make sure we work with iterators as well as pointers
|
||||
std::vector<int> v(first, first + sz);
|
||||
test_one_partial_sort(v.begin(), v.begin() + (sz / 2), v.end());
|
||||
test_one_partial_sort(v.begin(), v.begin() + (sz / 2), v.end(), std::greater<int>());
|
||||
}
|
||||
|
||||
|
||||
//// ===================================
|
||||
//// ==== indirect_nth_element_sort ====
|
||||
//// ===================================
|
||||
|
||||
template <typename Iter>
|
||||
void test_one_nth_element(Iter first, Iter nth, Iter last) {
|
||||
const size_t nthIdx = std::distance(first, nth);
|
||||
Permutation perm = boost::algorithm::indirect_nth_element(first, nth, last);
|
||||
BOOST_CHECK (is_a_permutation(perm, std::distance(first, last)));
|
||||
|
||||
for (size_t i = 0; i < nthIdx; ++i)
|
||||
BOOST_CHECK(!(first[perm[nthIdx]] < first[perm[i]])); // all items before the nth element are <= the nth element
|
||||
for (size_t i = nthIdx; i < std::distance(first, last); ++i)
|
||||
BOOST_CHECK(!(first[perm[i]] < first[perm[nthIdx]])); // all items before the nth element are >= the nth element
|
||||
|
||||
// Make a copy of the data, apply the permutation, and ensure that the result is correct.
|
||||
typedef std::vector<typename std::iterator_traits<Iter>::value_type> Vector;
|
||||
Vector v(first, last);
|
||||
boost::algorithm::apply_permutation(v.begin(), v.end(), perm.begin(), perm.end());
|
||||
|
||||
for (size_t i = 0; i < nthIdx; ++i)
|
||||
BOOST_CHECK(!(v[nthIdx] < v[i])); // all items before the nth element are <= the nth element
|
||||
for (size_t i = nthIdx; i < v.size(); ++i)
|
||||
BOOST_CHECK(!(v[i] < v[nthIdx])); // all items before the nth element are >= the nth element
|
||||
}
|
||||
|
||||
template <typename Iter, typename Comp>
|
||||
void test_one_nth_element(Iter first, Iter nth, Iter last, Comp comp) {
|
||||
const size_t nthIdx = std::distance(first, nth);
|
||||
|
||||
Permutation perm = boost::algorithm::indirect_nth_element(first, nth, last, comp);
|
||||
BOOST_CHECK (is_a_permutation(perm, std::distance(first, last)));
|
||||
for (size_t i = 0; i < nthIdx; ++i)
|
||||
BOOST_CHECK(!comp(first[perm[nthIdx]], first[perm[i]])); // all items before the nth element are <= the nth element
|
||||
for (size_t i = nthIdx; i < std::distance(first, last); ++i)
|
||||
BOOST_CHECK(!comp(first[perm[i]], first[perm[nthIdx]])); // all items before the nth element are >= the nth element
|
||||
|
||||
|
||||
// Make a copy of the data, apply the permutation, and ensure that the result is correct.
|
||||
typedef std::vector<typename std::iterator_traits<Iter>::value_type> Vector;
|
||||
Vector v(first, last);
|
||||
boost::algorithm::apply_permutation(v.begin(), v.end(), perm.begin(), perm.end());
|
||||
|
||||
for (size_t i = 0; i < nthIdx; ++i)
|
||||
BOOST_CHECK(!comp(v[nthIdx], v[i])); // all items before the nth element are <= the nth element
|
||||
for (size_t i = nthIdx; i < v.size(); ++i)
|
||||
BOOST_CHECK(!comp(v[i], v[nthIdx])); // all items before the nth element are >= the nth element
|
||||
}
|
||||
|
||||
|
||||
BOOST_AUTO_TEST_CASE(test_nth_element) {
|
||||
int num[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 10, 1, 2, 3, 4, 5 };
|
||||
const int sz = sizeof (num)/sizeof(num[0]);
|
||||
int *first = &num[0];
|
||||
int const *cFirst = &num[0];
|
||||
|
||||
// Test subsets
|
||||
for (size_t i = 0; i <= sz; ++i) {
|
||||
for (size_t j = 0; j < i; ++j) {
|
||||
test_one_nth_element(first, first + j, first + i);
|
||||
test_one_nth_element(first, first + j, first + i, std::greater<int>());
|
||||
|
||||
// test with constant inputs
|
||||
test_one_nth_element(cFirst, cFirst + j, cFirst + i);
|
||||
test_one_nth_element(cFirst, cFirst + j, cFirst + i, std::greater<int>());
|
||||
}
|
||||
}
|
||||
|
||||
// make sure we work with iterators as well as pointers
|
||||
std::vector<int> v(first, first + sz);
|
||||
test_one_nth_element(v.begin(), v.begin() + (sz / 2), v.end());
|
||||
test_one_nth_element(v.begin(), v.begin() + (sz / 2), v.end(), std::greater<int>());
|
||||
}
|
@ -112,7 +112,7 @@ void test_sequence1 () {
|
||||
std::vector<int> v, v1;
|
||||
|
||||
v.clear ();
|
||||
for ( int i = 5; i < 15; ++i )
|
||||
for ( std::size_t i = 5; i < 15; ++i )
|
||||
v.push_back ( i );
|
||||
v1 = v;
|
||||
BOOST_CHECK ( ba::is_permutation ( v.begin (), v.end (), v.begin ())); // better be a permutation of itself!
|
||||
|
@ -32,7 +32,7 @@ namespace {
|
||||
|
||||
// Check using iterators
|
||||
template<typename Container>
|
||||
void check_one_iter ( const Container &haystack, const std::string &needle, std::ptrdiff_t expected ) {
|
||||
void check_one_iter ( const Container &haystack, const std::string &needle, int expected ) {
|
||||
typedef typename Container::const_iterator iter_type;
|
||||
typedef typename std::pair<iter_type, iter_type> ret_type;
|
||||
typedef std::string::const_iterator pattern_type;
|
||||
@ -53,7 +53,7 @@ namespace {
|
||||
// iter_type it1r = ret1r.first;
|
||||
// iter_type it2 = ret2.first;
|
||||
// iter_type it3 = ret3.first;
|
||||
const std::ptrdiff_t dist = ret1.first == hEnd ? -1 : std::distance ( hBeg, ret1.first );
|
||||
const int dist = ret1.first == hEnd ? -1 : std::distance ( hBeg, ret1.first );
|
||||
|
||||
std::cout << "(Iterators) Pattern is " << needle.length () << ", haysstack is " << haystack.length () << " chars long; " << std::endl;
|
||||
try {
|
||||
@ -97,7 +97,7 @@ namespace {
|
||||
// Check using pointers
|
||||
// We're assuming that the container implements contiguous storage here.
|
||||
template<typename Container>
|
||||
void check_one_pointer ( const Container &haystack, const std::string &needle, std::ptrdiff_t expected ) {
|
||||
void check_one_pointer ( const Container &haystack, const std::string &needle, int expected ) {
|
||||
typedef const typename Container::value_type *ptr_type;
|
||||
typedef typename std::pair<ptr_type, ptr_type> ret_type;
|
||||
|
||||
@ -110,7 +110,7 @@ namespace {
|
||||
ret_type ret1 = ba::boyer_moore_search (hBeg, hEnd, nBeg, nEnd);
|
||||
ret_type ret2 = ba::boyer_moore_horspool_search (hBeg, hEnd, nBeg, nEnd);
|
||||
ret_type ret3 = ba::knuth_morris_pratt_search (hBeg, hEnd, nBeg, nEnd);
|
||||
const std::ptrdiff_t dist = ret1.first == hEnd ? -1 : std::distance ( hBeg, ret1.first );
|
||||
const int dist = ret1.first == hEnd ? -1 : std::distance ( hBeg, ret1.first );
|
||||
|
||||
std::cout << "(Pointers) Pattern is " << needle.length () << ", haysstack is " << haystack.length () << " chars long; " << std::endl;
|
||||
try {
|
||||
@ -147,7 +147,7 @@ namespace {
|
||||
|
||||
// Check using objects
|
||||
template<typename Container>
|
||||
void check_one_object ( const Container &haystack, const std::string &needle, std::ptrdiff_t expected ) {
|
||||
void check_one_object ( const Container &haystack, const std::string &needle, int expected ) {
|
||||
typedef typename Container::const_iterator iter_type;
|
||||
typedef typename std::pair<iter_type, iter_type> ret_type;
|
||||
typedef std::string::const_iterator pattern_type;
|
||||
@ -169,7 +169,7 @@ namespace {
|
||||
ret_type retr1r = bm_r (haystack);
|
||||
ret_type ret2 = bmh (hBeg, hEnd);
|
||||
ret_type ret3 = kmp (hBeg, hEnd);
|
||||
const std::ptrdiff_t dist = ret1.first == hEnd ? -1 : std::distance ( hBeg, ret1.first );
|
||||
const int dist = ret1.first == hEnd ? -1 : std::distance ( hBeg, ret1.first );
|
||||
|
||||
std::cout << "(Objects) Pattern is " << needle.length () << ", haysstack is " << haystack.length () << " chars long; " << std::endl;
|
||||
try {
|
||||
@ -224,7 +224,7 @@ namespace {
|
||||
|
||||
|
||||
template<typename Container>
|
||||
void check_one ( const Container &haystack, const std::string &needle, std::ptrdiff_t expected ) {
|
||||
void check_one ( const Container &haystack, const std::string &needle, int expected ) {
|
||||
check_one_iter ( haystack, needle, expected );
|
||||
check_one_pointer ( haystack, needle, expected );
|
||||
check_one_object ( haystack, needle, expected );
|
||||
|
@ -85,7 +85,7 @@ namespace {
|
||||
std::cout << std::endl;
|
||||
}
|
||||
|
||||
void check_one ( const vec &haystack, const vec &needle, std::ptrdiff_t expected ) {
|
||||
void check_one ( const vec &haystack, const vec &needle, int expected ) {
|
||||
std::size_t i;
|
||||
std::clock_t sTime;
|
||||
unsigned long stdDiff;
|
||||
@ -147,7 +147,7 @@ BOOST_AUTO_TEST_CASE( test_main )
|
||||
std::cout << "---- Middle -----" << std::endl;
|
||||
check_one ( c1, p1f, -2 ); // Don't know answer
|
||||
std::cout << "------ End ------" << std::endl;
|
||||
check_one ( c1, p1e, static_cast<std::ptrdiff_t>(c1.size() - p1e.size ()));
|
||||
check_one ( c1, p1e, c1.size() - p1e.size ());
|
||||
std::cout << "--- Not found ---" << std::endl;
|
||||
check_one ( c1, p1n, -1 ); // Not found
|
||||
}
|
||||
|
@ -85,7 +85,7 @@ namespace {
|
||||
std::cout << std::endl;
|
||||
}
|
||||
|
||||
void check_one ( const vec &haystack, const vec &needle, std::ptrdiff_t expected ) {
|
||||
void check_one ( const vec &haystack, const vec &needle, int expected ) {
|
||||
std::size_t i;
|
||||
std::clock_t sTime;
|
||||
unsigned long stdDiff;
|
||||
@ -147,7 +147,7 @@ BOOST_AUTO_TEST_CASE( test_main )
|
||||
std::cout << "---- Middle -----" << std::endl;
|
||||
check_one ( c1, p1f, -2 ); // Don't know answer
|
||||
std::cout << "------ End ------" << std::endl;
|
||||
check_one ( c1, p1e, static_cast<std::ptrdiff_t>(c1.size() - p1e.size ()));
|
||||
check_one ( c1, p1e, c1.size() - p1e.size ());
|
||||
std::cout << "--- Not found ---" << std::endl;
|
||||
check_one ( c1, p1n, -1 ); // Not found
|
||||
}
|
||||
|
@ -62,7 +62,7 @@ namespace {
|
||||
return retVal;
|
||||
}
|
||||
|
||||
void check_one ( const vec &haystack, const vec &needle, std::ptrdiff_t expected ) {
|
||||
void check_one ( const vec &haystack, const vec &needle, int expected ) {
|
||||
|
||||
std::pair<vec::const_iterator, vec::const_iterator> res;
|
||||
std::pair<vec::const_iterator, vec::const_iterator> exp; // the expected result
|
||||
@ -117,7 +117,7 @@ BOOST_AUTO_TEST_CASE( test_main )
|
||||
std::cout << "---- Middle -----" << std::endl;
|
||||
check_one ( c1, p1f, -2 ); // Don't know answer
|
||||
std::cout << "------ End ------" << std::endl;
|
||||
check_one ( c1, p1e, static_cast<std::ptrdiff_t>(c1.size() - p1e.size ()));
|
||||
check_one ( c1, p1e, c1.size() - p1e.size ());
|
||||
std::cout << "--- Not found ---" << std::endl;
|
||||
check_one ( c1, p1n, -1 ); // Not found
|
||||
}
|
||||
|
Reference in New Issue
Block a user