forked from OSchip/llvm-project
- Add hasNItemsOrLess and container variants of hasNItems, hasNItemsOrMore, and hasNItemsOrLess
- Fixed a bug in hasNItems() - Extend the STLExtras unit test to test hasSingleElement() and hasNItems() and friends. Differential Revision: https://reviews.llvm.org/D82232
This commit is contained in:
parent
d335c1317b
commit
0d9726bc3d
|
@ -267,10 +267,10 @@ constexpr bool empty(const T &RangeOrContainer) {
|
||||||
return adl_begin(RangeOrContainer) == adl_end(RangeOrContainer);
|
return adl_begin(RangeOrContainer) == adl_end(RangeOrContainer);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns true of the given range only contains a single element.
|
/// Returns true if the given container only contains a single element.
|
||||||
template <typename ContainerTy> bool hasSingleElement(ContainerTy &&c) {
|
template <typename ContainerTy> bool hasSingleElement(ContainerTy &&C) {
|
||||||
auto it = std::begin(c), e = std::end(c);
|
auto B = std::begin(C), E = std::end(C);
|
||||||
return it != e && std::next(it) == e;
|
return B != E && std::next(B) == E;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Return a range covering \p RangeOrContainer with the first N elements
|
/// Return a range covering \p RangeOrContainer with the first N elements
|
||||||
|
@ -1919,12 +1919,15 @@ bool hasNItems(
|
||||||
return false; // Too few.
|
return false; // Too few.
|
||||||
N -= ShouldBeCounted(*Begin);
|
N -= ShouldBeCounted(*Begin);
|
||||||
}
|
}
|
||||||
return Begin == End;
|
for (; Begin != End; ++Begin)
|
||||||
|
if (ShouldBeCounted(*Begin))
|
||||||
|
return false; // Too many.
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Return true if the sequence [Begin, End) has N or more items. Runs in O(N)
|
/// Return true if the sequence [Begin, End) has N or more items. Runs in O(N)
|
||||||
/// time. Not meant for use with random-access iterators.
|
/// time. Not meant for use with random-access iterators.
|
||||||
/// Can optionally take a predicate to filter lazily some items.
|
/// Can optionally take a predicate to lazily filter some items.
|
||||||
template<typename IterTy,
|
template<typename IterTy,
|
||||||
typename Pred = bool (*)(const decltype(*std::declval<IterTy>()) &)>
|
typename Pred = bool (*)(const decltype(*std::declval<IterTy>()) &)>
|
||||||
bool hasNItemsOrMore(
|
bool hasNItemsOrMore(
|
||||||
|
@ -1944,6 +1947,36 @@ bool hasNItemsOrMore(
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns true if the sequence [Begin, End) has N or less items. Can
|
||||||
|
/// optionally take a predicate to lazily filter some items.
|
||||||
|
template <typename IterTy,
|
||||||
|
typename Pred = bool (*)(const decltype(*std::declval<IterTy>()) &)>
|
||||||
|
bool hasNItemsOrLess(
|
||||||
|
IterTy &&Begin, IterTy &&End, unsigned N,
|
||||||
|
Pred &&ShouldBeCounted = [](const decltype(*std::declval<IterTy>()) &) {
|
||||||
|
return true;
|
||||||
|
}) {
|
||||||
|
assert(N != std::numeric_limits<unsigned>::max());
|
||||||
|
return !hasNItemsOrMore(Begin, End, N + 1, ShouldBeCounted);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns true if the given container has exactly N items
|
||||||
|
template <typename ContainerTy> bool hasNItems(ContainerTy &&C, unsigned N) {
|
||||||
|
return hasNItems(std::begin(C), std::end(C), N);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns true if the given container has N or more items
|
||||||
|
template <typename ContainerTy>
|
||||||
|
bool hasNItemsOrMore(ContainerTy &&C, unsigned N) {
|
||||||
|
return hasNItemsOrMore(std::begin(C), std::end(C), N);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns true if the given container has N or less items
|
||||||
|
template <typename ContainerTy>
|
||||||
|
bool hasNItemsOrLess(ContainerTy &&C, unsigned N) {
|
||||||
|
return hasNItemsOrLess(std::begin(C), std::end(C), N);
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns a raw pointer that represents the same address as the argument.
|
/// Returns a raw pointer that represents the same address as the argument.
|
||||||
///
|
///
|
||||||
/// This implementation can be removed once we move to C++20 where it's defined
|
/// This implementation can be removed once we move to C++20 where it's defined
|
||||||
|
|
|
@ -490,4 +490,82 @@ TEST(STLExtrasTest, partition_point) {
|
||||||
EXPECT_EQ(V.end(), partition_point(V, [](unsigned X) { return X < 50; }));
|
EXPECT_EQ(V.end(), partition_point(V, [](unsigned X) { return X < 50; }));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST(STLExtrasTest, hasSingleElement) {
|
||||||
|
const std::vector<int> V0 = {}, V1 = {1}, V2 = {1, 2};
|
||||||
|
const std::vector<int> V10(10);
|
||||||
|
|
||||||
|
EXPECT_EQ(hasSingleElement(V0), false);
|
||||||
|
EXPECT_EQ(hasSingleElement(V1), true);
|
||||||
|
EXPECT_EQ(hasSingleElement(V2), false);
|
||||||
|
EXPECT_EQ(hasSingleElement(V10), false);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(STLExtrasTest, hasNItems) {
|
||||||
|
const std::list<int> V0 = {}, V1 = {1}, V2 = {1, 2};
|
||||||
|
const std::list<int> V3 = {1, 3, 5};
|
||||||
|
|
||||||
|
EXPECT_TRUE(hasNItems(V0, 0));
|
||||||
|
EXPECT_FALSE(hasNItems(V0, 2));
|
||||||
|
EXPECT_TRUE(hasNItems(V1, 1));
|
||||||
|
EXPECT_FALSE(hasNItems(V1, 2));
|
||||||
|
|
||||||
|
EXPECT_TRUE(hasNItems(V3.begin(), V3.end(), 3, [](int x) { return x < 10; }));
|
||||||
|
EXPECT_TRUE(hasNItems(V3.begin(), V3.end(), 0, [](int x) { return x > 10; }));
|
||||||
|
EXPECT_TRUE(hasNItems(V3.begin(), V3.end(), 2, [](int x) { return x < 5; }));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(STLExtras, hasNItemsOrMore) {
|
||||||
|
const std::list<int> V0 = {}, V1 = {1}, V2 = {1, 2};
|
||||||
|
const std::list<int> V3 = {1, 3, 5};
|
||||||
|
|
||||||
|
EXPECT_TRUE(hasNItemsOrMore(V1, 1));
|
||||||
|
EXPECT_FALSE(hasNItemsOrMore(V1, 2));
|
||||||
|
|
||||||
|
EXPECT_TRUE(hasNItemsOrMore(V2, 1));
|
||||||
|
EXPECT_TRUE(hasNItemsOrMore(V2, 2));
|
||||||
|
EXPECT_FALSE(hasNItemsOrMore(V2, 3));
|
||||||
|
|
||||||
|
EXPECT_TRUE(hasNItemsOrMore(V3, 3));
|
||||||
|
EXPECT_FALSE(hasNItemsOrMore(V3, 4));
|
||||||
|
|
||||||
|
EXPECT_TRUE(
|
||||||
|
hasNItemsOrMore(V3.begin(), V3.end(), 3, [](int x) { return x < 10; }));
|
||||||
|
EXPECT_FALSE(
|
||||||
|
hasNItemsOrMore(V3.begin(), V3.end(), 3, [](int x) { return x > 10; }));
|
||||||
|
EXPECT_TRUE(
|
||||||
|
hasNItemsOrMore(V3.begin(), V3.end(), 2, [](int x) { return x < 5; }));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(STLExtras, hasNItemsOrLess) {
|
||||||
|
const std::list<int> V0 = {}, V1 = {1}, V2 = {1, 2};
|
||||||
|
const std::list<int> V3 = {1, 3, 5};
|
||||||
|
|
||||||
|
EXPECT_TRUE(hasNItemsOrLess(V0, 0));
|
||||||
|
EXPECT_TRUE(hasNItemsOrLess(V0, 1));
|
||||||
|
EXPECT_TRUE(hasNItemsOrLess(V0, 2));
|
||||||
|
|
||||||
|
EXPECT_FALSE(hasNItemsOrLess(V1, 0));
|
||||||
|
EXPECT_TRUE(hasNItemsOrLess(V1, 1));
|
||||||
|
EXPECT_TRUE(hasNItemsOrLess(V1, 2));
|
||||||
|
|
||||||
|
EXPECT_FALSE(hasNItemsOrLess(V2, 0));
|
||||||
|
EXPECT_FALSE(hasNItemsOrLess(V2, 1));
|
||||||
|
EXPECT_TRUE(hasNItemsOrLess(V2, 2));
|
||||||
|
EXPECT_TRUE(hasNItemsOrLess(V2, 3));
|
||||||
|
|
||||||
|
EXPECT_FALSE(hasNItemsOrLess(V3, 0));
|
||||||
|
EXPECT_FALSE(hasNItemsOrLess(V3, 1));
|
||||||
|
EXPECT_FALSE(hasNItemsOrLess(V3, 2));
|
||||||
|
EXPECT_TRUE(hasNItemsOrLess(V3, 3));
|
||||||
|
EXPECT_TRUE(hasNItemsOrLess(V3, 4));
|
||||||
|
|
||||||
|
EXPECT_TRUE(
|
||||||
|
hasNItemsOrLess(V3.begin(), V3.end(), 1, [](int x) { return x == 1; }));
|
||||||
|
EXPECT_TRUE(
|
||||||
|
hasNItemsOrLess(V3.begin(), V3.end(), 2, [](int x) { return x < 5; }));
|
||||||
|
EXPECT_TRUE(
|
||||||
|
hasNItemsOrLess(V3.begin(), V3.end(), 5, [](int x) { return x < 5; }));
|
||||||
|
EXPECT_FALSE(
|
||||||
|
hasNItemsOrLess(V3.begin(), V3.end(), 2, [](int x) { return x < 10; }));
|
||||||
|
}
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
Loading…
Reference in New Issue