//===----------------------------------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // UNSUPPORTED: c++03, c++11, c++14 // // template // SampleIterator sample(PopulationIterator first, PopulationIterator last, // SampleIterator out, Distance n, // UniformRandomNumberGenerator &&g); #include #include #include #include #include #include "test_iterators.h" #include "test_macros.h" struct ReservoirSampleExpectations { enum { os = 4 }; static int oa1[os]; static int oa2[os]; }; int ReservoirSampleExpectations::oa1[] = {10, 5, 9, 4}; int ReservoirSampleExpectations::oa2[] = {5, 2, 10, 4}; struct SelectionSampleExpectations { enum { os = 4 }; static int oa1[os]; static int oa2[os]; }; int SelectionSampleExpectations::oa1[] = {1, 4, 6, 7}; int SelectionSampleExpectations::oa2[] = {1, 2, 6, 8}; template struct TestExpectations : public SelectionSampleExpectations {}; template <> struct TestExpectations : public ReservoirSampleExpectations {}; template class PopulationIteratorType, class PopulationItem, template class SampleIteratorType, class SampleItem> void test() { typedef PopulationIteratorType PopulationIterator; typedef SampleIteratorType SampleIterator; PopulationItem ia[] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10}; const unsigned is = sizeof(ia) / sizeof(ia[0]); typedef TestExpectations::iterator_category> Expectations; const unsigned os = Expectations::os; SampleItem oa[os]; const int *oa1 = Expectations::oa1; ((void)oa1); // Prevent unused warning const int *oa2 = Expectations::oa2; ((void)oa2); // Prevent unused warning std::minstd_rand g; SampleIterator end; end = std::sample(PopulationIterator(ia), PopulationIterator(ia + is), SampleIterator(oa), os, g); assert(static_cast(end.base() - oa) == std::min(os, is)); // sample() is deterministic but non-reproducible; // its results can vary between implementations. LIBCPP_ASSERT(std::equal(oa, oa + os, oa1)); end = std::sample(PopulationIterator(ia), PopulationIterator(ia + is), SampleIterator(oa), os, std::move(g)); assert(static_cast(end.base() - oa) == std::min(os, is)); LIBCPP_ASSERT(std::equal(oa, oa + os, oa2)); } template class PopulationIteratorType, class PopulationItem, template class SampleIteratorType, class SampleItem> void test_empty_population() { typedef PopulationIteratorType PopulationIterator; typedef SampleIteratorType SampleIterator; PopulationItem ia[] = {42}; const unsigned os = 4; SampleItem oa[os]; std::minstd_rand g; SampleIterator end = std::sample(PopulationIterator(ia), PopulationIterator(ia), SampleIterator(oa), os, g); assert(end.base() == oa); } template class PopulationIteratorType, class PopulationItem, template class SampleIteratorType, class SampleItem> void test_empty_sample() { typedef PopulationIteratorType PopulationIterator; typedef SampleIteratorType SampleIterator; PopulationItem ia[] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10}; const unsigned is = sizeof(ia) / sizeof(ia[0]); SampleItem oa[1]; std::minstd_rand g; SampleIterator end = std::sample(PopulationIterator(ia), PopulationIterator(ia + is), SampleIterator(oa), 0, g); assert(end.base() == oa); } template class PopulationIteratorType, class PopulationItem, template class SampleIteratorType, class SampleItem> void test_small_population() { // The population size is less than the sample size. typedef PopulationIteratorType PopulationIterator; typedef SampleIteratorType SampleIterator; PopulationItem ia[] = {1, 2, 3, 4, 5}; const unsigned is = sizeof(ia) / sizeof(ia[0]); const unsigned os = 8; SampleItem oa[os]; const SampleItem oa1[] = {1, 2, 3, 4, 5}; std::minstd_rand g; SampleIterator end; end = std::sample(PopulationIterator(ia), PopulationIterator(ia + is), SampleIterator(oa), os, g); assert(static_cast(end.base() - oa) == std::min(os, is)); typedef typename std::iterator_traits::iterator_category PopulationCategory; if (std::is_base_of::value) { assert(std::equal(oa, end.base(), oa1)); } else { assert(std::is_permutation(oa, end.base(), oa1)); } } int main(int, char**) { test(); test(); test(); test(); test(); test(); test_empty_population(); test_empty_population(); test_empty_population(); test_empty_sample(); test_empty_sample(); test_empty_sample(); test_small_population(); test_small_population(); test_small_population(); return 0; }