Skip to content

Commit c13c27a

Browse files
Abseil Teamderekmauro
Abseil Team
authored andcommitted
Googletest export
Change Matcher<T> to allow binding an implementation by value directly: - Drop the requirement of MatcherInterface. Doing manual type erasure avoid extra layers in many cases. - Avoid the adaptor for `MatcherInterface<T>` and `MatcherInterface<const T&>` mismatch. - Use a small object optimization when possible. This makes things like `_` and `Eq(1)` really cheap and do not require memory allocations. - Migrate some matchers to the new model to speed them up and to test the new framework. More matchers to come in future changes. PiperOrigin-RevId: 350580998
1 parent 4892835 commit c13c27a

File tree

4 files changed

+388
-170
lines changed

4 files changed

+388
-170
lines changed

docs/gmock_cook_book.md

+113-66
Original file line numberDiff line numberDiff line change
@@ -1315,32 +1315,30 @@ how you can define a matcher to do it:
13151315

13161316
```cpp
13171317
using ::testing::Matcher;
1318-
using ::testing::MatcherInterface;
1319-
using ::testing::MatchResultListener;
13201318

1321-
class BarPlusBazEqMatcher : public MatcherInterface<const Foo&> {
1319+
class BarPlusBazEqMatcher {
13221320
public:
13231321
explicit BarPlusBazEqMatcher(int expected_sum)
13241322
: expected_sum_(expected_sum) {}
13251323

13261324
bool MatchAndExplain(const Foo& foo,
1327-
MatchResultListener* /* listener */) const override {
1325+
std::ostream* /* listener */) const {
13281326
return (foo.bar() + foo.baz()) == expected_sum_;
13291327
}
13301328

1331-
void DescribeTo(std::ostream* os) const override {
1332-
*os << "bar() + baz() equals " << expected_sum_;
1329+
void DescribeTo(std::ostream& os) const {
1330+
os << "bar() + baz() equals " << expected_sum_;
13331331
}
13341332

1335-
void DescribeNegationTo(std::ostream* os) const override {
1336-
*os << "bar() + baz() does not equal " << expected_sum_;
1333+
void DescribeNegationTo(std::ostream& os) const {
1334+
os << "bar() + baz() does not equal " << expected_sum_;
13371335
}
13381336
private:
13391337
const int expected_sum_;
13401338
};
13411339

13421340
Matcher<const Foo&> BarPlusBazEq(int expected_sum) {
1343-
return MakeMatcher(new BarPlusBazEqMatcher(expected_sum));
1341+
return BarPlusBazEqMatcher(expected_sum);
13441342
}
13451343

13461344
...
@@ -3535,51 +3533,39 @@ MATCHER_P2(Blah, a, b, description_string_2) { ... }
35353533
```
35363534

35373535
While it's tempting to always use the `MATCHER*` macros when defining a new
3538-
matcher, you should also consider implementing `MatcherInterface` or using
3539-
`MakePolymorphicMatcher()` instead (see the recipes that follow), especially if
3540-
you need to use the matcher a lot. While these approaches require more work,
3541-
they give you more control on the types of the value being matched and the
3542-
matcher parameters, which in general leads to better compiler error messages
3543-
that pay off in the long run. They also allow overloading matchers based on
3544-
parameter types (as opposed to just based on the number of parameters).
3536+
matcher, you should also consider implementing the matcher interface directly
3537+
instead (see the recipes that follow), especially if you need to use the matcher
3538+
a lot. While these approaches require more work, they give you more control on
3539+
the types of the value being matched and the matcher parameters, which in
3540+
general leads to better compiler error messages that pay off in the long run.
3541+
They also allow overloading matchers based on parameter types (as opposed to
3542+
just based on the number of parameters).
35453543

35463544
### Writing New Monomorphic Matchers
35473545

3548-
A matcher of argument type `T` implements `::testing::MatcherInterface<T>` and
3549-
does two things: it tests whether a value of type `T` matches the matcher, and
3550-
can describe what kind of values it matches. The latter ability is used for
3546+
A matcher of argument type `T` implements the matcher interface for `T` and does
3547+
two things: it tests whether a value of type `T` matches the matcher, and can
3548+
describe what kind of values it matches. The latter ability is used for
35513549
generating readable error messages when expectations are violated.
35523550

3553-
The interface looks like this:
3551+
A matcher of `T` must declare a typedef like:
35543552

35553553
```cpp
3556-
class MatchResultListener {
3557-
public:
3558-
...
3559-
// Streams x to the underlying ostream; does nothing if the ostream
3560-
// is NULL.
3561-
template <typename T>
3562-
MatchResultListener& operator<<(const T& x);
3563-
3564-
// Returns the underlying ostream.
3565-
std::ostream* stream();
3566-
};
3567-
3568-
template <typename T>
3569-
class MatcherInterface {
3570-
public:
3571-
virtual ~MatcherInterface();
3554+
using is_gtest_matcher = void;
3555+
```
35723556

3573-
// Returns true if and only if the matcher matches x; also explains the match
3574-
// result to 'listener'.
3575-
virtual bool MatchAndExplain(T x, MatchResultListener* listener) const = 0;
3557+
and supports the following operations:
35763558

3577-
// Describes this matcher to an ostream.
3578-
virtual void DescribeTo(std::ostream* os) const = 0;
3559+
```cpp
3560+
// Match a value and optionally explain into an ostream.
3561+
bool matched = matcher.MatchAndExplain(value, maybe_os);
3562+
// where `value` is of type `T` and
3563+
// `maybe_os` is of type `std::ostream*`, where it can be null if the caller
3564+
// is not interested in there textual explanation.
35793565

3580-
// Describes the negation of this matcher to an ostream.
3581-
virtual void DescribeNegationTo(std::ostream* os) const;
3582-
};
3566+
matcher.DescribeTo(os);
3567+
matcher.DescribeNegationTo(os);
3568+
// where `os` is of type `std::ostream*`.
35833569
```
35843570

35853571
If you need a custom matcher but `Truly()` is not a good option (for example,
@@ -3594,46 +3580,41 @@ For example, you can define a matcher to test whether an `int` is divisible by 7
35943580
and then use it like this:
35953581

35963582
```cpp
3597-
using ::testing::MakeMatcher;
35983583
using ::testing::Matcher;
3599-
using ::testing::MatcherInterface;
3600-
using ::testing::MatchResultListener;
36013584

3602-
class DivisibleBy7Matcher : public MatcherInterface<int> {
3585+
class DivisibleBy7Matcher {
36033586
public:
3604-
bool MatchAndExplain(int n,
3605-
MatchResultListener* /* listener */) const override {
3587+
bool MatchAndExplain(int n, std::ostream*) const {
36063588
return (n % 7) == 0;
36073589
}
36083590

3609-
void DescribeTo(std::ostream* os) const override {
3591+
void DescribeTo(std::ostream* os) const {
36103592
*os << "is divisible by 7";
36113593
}
36123594

3613-
void DescribeNegationTo(std::ostream* os) const override {
3595+
void DescribeNegationTo(std::ostream* os) const {
36143596
*os << "is not divisible by 7";
36153597
}
36163598
};
36173599

36183600
Matcher<int> DivisibleBy7() {
3619-
return MakeMatcher(new DivisibleBy7Matcher);
3601+
return DivisibleBy7Matcher();
36203602
}
36213603

36223604
...
36233605
EXPECT_CALL(foo, Bar(DivisibleBy7()));
36243606
```
36253607

36263608
You may improve the matcher message by streaming additional information to the
3627-
`listener` argument in `MatchAndExplain()`:
3609+
`os` argument in `MatchAndExplain()`:
36283610

36293611
```cpp
3630-
class DivisibleBy7Matcher : public MatcherInterface<int> {
3612+
class DivisibleBy7Matcher {
36313613
public:
3632-
bool MatchAndExplain(int n,
3633-
MatchResultListener* listener) const override {
3614+
bool MatchAndExplain(int n, std::ostream* os) const {
36343615
const int remainder = n % 7;
3635-
if (remainder != 0) {
3636-
*listener << "the remainder is " << remainder;
3616+
if (remainder != 0 && os != nullptr) {
3617+
*os << "the remainder is " << remainder;
36373618
}
36383619
return remainder == 0;
36393620
}
@@ -3651,13 +3632,79 @@ Expected: is divisible by 7
36513632

36523633
### Writing New Polymorphic Matchers
36533634

3654-
You've learned how to write your own matchers in the previous recipe. Just one
3655-
problem: a matcher created using `MakeMatcher()` only works for one particular
3656-
type of arguments. If you want a *polymorphic* matcher that works with arguments
3657-
of several types (for instance, `Eq(x)` can be used to match a *`value`* as long
3658-
as `value == x` compiles -- *`value`* and `x` don't have to share the same
3659-
type), you can learn the trick from `testing/base/public/gmock-matchers.h` but
3660-
it's a bit involved.
3635+
Expanding what we learned above to *polymorphic* matchers is now just as simple
3636+
as adding templates in the right place.
3637+
3638+
```cpp
3639+
3640+
class NotNullMatcher {
3641+
public:
3642+
// To implement a polymorphic matcher, we just need to make MatchAndExplain a
3643+
// template on its first argument.
3644+
3645+
// In this example, we want to use NotNull() with any pointer, so
3646+
// MatchAndExplain() accepts a pointer of any type as its first argument.
3647+
// In general, you can define MatchAndExplain() as an ordinary method or
3648+
// a method template, or even overload it.
3649+
template <typename T>
3650+
bool MatchAndExplain(T* p, std::ostream*) const {
3651+
return p != nullptr;
3652+
}
3653+
3654+
// Describes the property of a value matching this matcher.
3655+
void DescribeTo(std::ostream& os) const { *os << "is not NULL"; }
3656+
3657+
// Describes the property of a value NOT matching this matcher.
3658+
void DescribeNegationTo(std::ostream* os) const { *os << "is NULL"; }
3659+
};
3660+
3661+
NotNullMatcher NotNull() {
3662+
return NotNullMatcher();
3663+
}
3664+
3665+
...
3666+
3667+
EXPECT_CALL(foo, Bar(NotNull())); // The argument must be a non-NULL pointer.
3668+
```
3669+
3670+
### Legacy Matcher Implementation
3671+
3672+
Defining matchers used to be somewhat more complicated, in which it required
3673+
several supporting classes and virtual functions. To implement a matcher for
3674+
type `T` using the legacy API you have to derive from `MatcherInterface<T>` and
3675+
call `MakeMatcher` to construct the object.
3676+
3677+
The interface looks like this:
3678+
3679+
```cpp
3680+
class MatchResultListener {
3681+
public:
3682+
...
3683+
// Streams x to the underlying ostream; does nothing if the ostream
3684+
// is NULL.
3685+
template <typename T>
3686+
MatchResultListener& operator<<(const T& x);
3687+
3688+
// Returns the underlying ostream.
3689+
std::ostream* stream();
3690+
};
3691+
3692+
template <typename T>
3693+
class MatcherInterface {
3694+
public:
3695+
virtual ~MatcherInterface();
3696+
3697+
// Returns true if and only if the matcher matches x; also explains the match
3698+
// result to 'listener'.
3699+
virtual bool MatchAndExplain(T x, MatchResultListener* listener) const = 0;
3700+
3701+
// Describes this matcher to an ostream.
3702+
virtual void DescribeTo(std::ostream* os) const = 0;
3703+
3704+
// Describes the negation of this matcher to an ostream.
3705+
virtual void DescribeNegationTo(std::ostream* os) const;
3706+
};
3707+
```
36613708

36623709
Fortunately, most of the time you can define a polymorphic matcher easily with
36633710
the help of `MakePolymorphicMatcher()`. Here's how you can define `NotNull()` as

googlemock/include/gmock/gmock-matchers.h

+19-21
Original file line numberDiff line numberDiff line change
@@ -735,31 +735,25 @@ OutIter TransformTupleValues(Func f, const Tuple& t, OutIter out) {
735735
return TransformTupleValuesHelper<Tuple, Func, OutIter>::Run(f, t, out);
736736
}
737737

738-
// Implements A<T>().
739-
template <typename T>
740-
class AnyMatcherImpl : public MatcherInterface<const T&> {
741-
public:
742-
bool MatchAndExplain(const T& /* x */,
743-
MatchResultListener* /* listener */) const override {
744-
return true;
745-
}
746-
void DescribeTo(::std::ostream* os) const override { *os << "is anything"; }
747-
void DescribeNegationTo(::std::ostream* os) const override {
748-
// This is mostly for completeness' safe, as it's not very useful
749-
// to write Not(A<bool>()). However we cannot completely rule out
750-
// such a possibility, and it doesn't hurt to be prepared.
751-
*os << "never matches";
752-
}
753-
};
754-
755738
// Implements _, a matcher that matches any value of any
756739
// type. This is a polymorphic matcher, so we need a template type
757740
// conversion operator to make it appearing as a Matcher<T> for any
758741
// type T.
759742
class AnythingMatcher {
760743
public:
744+
using is_gtest_matcher = void;
745+
761746
template <typename T>
762-
operator Matcher<T>() const { return A<T>(); }
747+
bool MatchAndExplain(const T& /* x */, std::ostream* /* listener */) const {
748+
return true;
749+
}
750+
void DescribeTo(std::ostream* os) const { *os << "is anything"; }
751+
void DescribeNegationTo(::std::ostream* os) const {
752+
// This is mostly for completeness' sake, as it's not very useful
753+
// to write Not(A<bool>()). However we cannot completely rule out
754+
// such a possibility, and it doesn't hurt to be prepared.
755+
*os << "never matches";
756+
}
763757
};
764758

765759
// Implements the polymorphic IsNull() matcher, which matches any raw or smart
@@ -3443,7 +3437,9 @@ class UnorderedElementsAreMatcherImpl
34433437
: UnorderedElementsAreMatcherImplBase(matcher_flags) {
34443438
for (; first != last; ++first) {
34453439
matchers_.push_back(MatcherCast<const Element&>(*first));
3446-
matcher_describers().push_back(matchers_.back().GetDescriber());
3440+
}
3441+
for (const auto& m : matchers_) {
3442+
matcher_describers().push_back(m.GetDescriber());
34473443
}
34483444
}
34493445

@@ -4068,12 +4064,14 @@ const internal::AnythingMatcher _ = {};
40684064
// Creates a matcher that matches any value of the given type T.
40694065
template <typename T>
40704066
inline Matcher<T> A() {
4071-
return Matcher<T>(new internal::AnyMatcherImpl<T>());
4067+
return _;
40724068
}
40734069

40744070
// Creates a matcher that matches any value of the given type T.
40754071
template <typename T>
4076-
inline Matcher<T> An() { return A<T>(); }
4072+
inline Matcher<T> An() {
4073+
return _;
4074+
}
40774075

40784076
template <typename T, typename M>
40794077
Matcher<T> internal::MatcherCastImpl<T, M>::CastImpl(

googlemock/test/gmock-matchers_test.cc

+1-1
Original file line numberDiff line numberDiff line change
@@ -410,7 +410,7 @@ TEST(StringMatcherTest,
410410
// MatcherInterface* without requiring the user to explicitly
411411
// write the type.
412412
TEST(MakeMatcherTest, ConstructsMatcherFromMatcherInterface) {
413-
const MatcherInterface<int>* dummy_impl = nullptr;
413+
const MatcherInterface<int>* dummy_impl = new EvenMatcherImpl;
414414
Matcher<int> m = MakeMatcher(dummy_impl);
415415
}
416416

0 commit comments

Comments
 (0)