//===----------------------------------------------------------------------===// // // The LLVM Compiler Infrastructure // // This file is dual licensed under the MIT and the University of Illinois Open // Source Licenses. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // // template // SampleIterator sample(PopulationIterator first, PopulationIterator last, // SampleIterator out, Distance n, // UniformRandomNumberGenerator &&g); #include #include #include #include "test_iterators.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; const int *oa2 = Expectations::oa2; std::minstd_rand g; SampleIterator end; end = std::experimental::sample(PopulationIterator(ia), PopulationIterator(ia + is), SampleIterator(oa), os, g); assert(&*end - oa == std::min(os, is)); assert(std::equal(oa, oa + os, oa1)); end = std::experimental::sample(PopulationIterator(ia), PopulationIterator(ia + is), SampleIterator(oa), os, g); assert(&*end - oa == std::min(os, is)); 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::experimental::sample(PopulationIterator(ia), PopulationIterator(ia), SampleIterator(oa), os, g); assert(&*end == 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::experimental::sample(PopulationIterator(ia), PopulationIterator(ia + is), SampleIterator(oa), 0, g); assert(&*end == 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::experimental::sample(PopulationIterator(ia), PopulationIterator(ia + is), SampleIterator(oa), os, g); assert(&*end - oa == std::min(os, is)); assert(std::equal(oa, &*end, oa1)); } int main() { 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(); }