日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 人文社科 > 生活经验 >内容正文

生活经验

Google Mock(Gmock)简单使用和源码分析——源码分析

發布時間:2023/11/27 生活经验 41 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Google Mock(Gmock)简单使用和源码分析——源码分析 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

源碼分析

? ? ? ? 通過《Google Mock(Gmock)簡單使用和源碼分析——簡單使用》中的例子,我們發現被mock的相關方法在mock類中已經被重新實現了,否則它們也不會按照我們的期待的行為執行。我們通過閱讀源碼,來分析整個過程的實現邏輯。(轉載請指明出于breaksoftware的csdn博客)

MOCK_METHOD系列宏

? ? ? ? 首先我們以MOCK_METHOD0為例

#define MOCK_METHOD0(m, ...) GMOCK_METHOD0_(, , , m, __VA_ARGS__)

? ? ? ? 可以看到它實際上封裝了GMOCK_METHOD0_。我們在介紹GMOCK_METHOD0_之前,還可以看到其他無參數的宏

#define MOCK_CONST_METHOD0(m, ...) GMOCK_METHOD0_(, const, , m, __VA_ARGS__)
……
#define MOCK_METHOD0_T(m, ...) GMOCK_METHOD0_(typename, , , m, __VA_ARGS__)
……
#define MOCK_CONST_METHOD0_T(m, ...) \GMOCK_METHOD0_(typename, const, , m, __VA_ARGS__)
……
#define MOCK_METHOD0_WITH_CALLTYPE(ct, m, ...) \GMOCK_METHOD0_(, , ct, m, __VA_ARGS__)
……
#define MOCK_CONST_METHOD0_WITH_CALLTYPE(ct, m, ...) \GMOCK_METHOD0_(, const, ct, m, __VA_ARGS__)
……
#define MOCK_METHOD0_T_WITH_CALLTYPE(ct, m, ...) \GMOCK_METHOD0_(typename, , ct, m, __VA_ARGS__)
……
#define MOCK_CONST_METHOD0_T_WITH_CALLTYPE(ct, m, ...) \GMOCK_METHOD0_(typename, const, ct, m, __VA_ARGS__)
……

? ? ? ? 這些無參數的宏宏都是基于GMOCK_METHOD0_實現的,它們的差別只是不同參數的組合。這兒要列出它們是因為GMOCK_METHOD0_的定義比較晦澀,通過這些醒目的定義,我們將會發現其各個參數的作用。

// INTERNAL IMPLEMENTATION - DON'T USE IN USER CODE!!!
#define GMOCK_METHOD0_(tn, constness, ct, Method, ...) \GMOCK_RESULT_(tn, __VA_ARGS__) ct Method( \) constness { \GTEST_COMPILE_ASSERT_((::testing::tuple_size<                          \tn ::testing::internal::Function<__VA_ARGS__>::ArgumentTuple>::value \== 0), \this_method_does_not_take_0_arguments); \GMOCK_MOCKER_(0, constness, Method).SetOwnerAndName(this, #Method); \return GMOCK_MOCKER_(0, constness, Method).Invoke(); \} \::testing::MockSpec<__VA_ARGS__>& \gmock_##Method() constness { \GMOCK_MOCKER_(0, constness, Method).RegisterOwner(this); \return GMOCK_MOCKER_(0, constness, Method).With(); \} \mutable ::testing::FunctionMocker<__VA_ARGS__> GMOCK_MOCKER_(0, constness, \Method)

? ? ? ? 上面宏中的tn可以通過之前的調用發現其應該是typename這類用于定義模板的關鍵字。constness表示mock的方法是不是const類型的。ct是調用約定,比如我們在windows程序里經常見到的STDMETHODCALLTYPE。Method是被mock的函數名。不定參數則是函數指針類型。這兒比較有意思的是不定參數,因為作為一個框架,它需要支持各種類型的函數,而我們不可能把所有類型一一進行羅列。這個時候我們就可以使用不定參數來解決這個問題。
? ? ? ? 我們先總覽一下GMOCK_METHOD0_的實現。上述代碼第17行定義了一個具有mutable屬性的變量,之所以使用mutable是因為它可能會被使用在const類型的函數中,然而該對象的方法并不一定是const的。這個參數的名稱使用GMOCK_MOCKER_宏組裝

#define GMOCK_MOCKER_(arity, constness, Method) \GTEST_CONCAT_TOKEN_(gmock##constness##arity##_##Method##_, __LINE__)#define GTEST_CONCAT_TOKEN_(foo, bar) GTEST_CONCAT_TOKEN_IMPL_(foo, bar)
#define GTEST_CONCAT_TOKEN_IMPL_(foo, bar) foo ## bar

? ? ? ? 可以見得該參數名包括了gmock關鍵字、是否有const屬性、參數個數、方法名已經所在的行號組成。這樣就盡可能的保證該變量在同一個文件中的唯一性。

? ? ? ? 該變量的類型是一個以函數類型為模板參數的對象,其模板類的定義是

template <typename R>
class FunctionMocker<R()> : publicinternal::FunctionMockerBase<R()> {public:typedef R F();typedef typename internal::Function<F>::ArgumentTuple ArgumentTuple;MockSpec<F>& With() {return this->current_spec();}R Invoke() {// Even though gcc and MSVC don't enforce it, 'this->' is required// by the C++ standard [14.6.4] here, as the base class type is// dependent on the template argument (and thus shouldn't be// looked into when resolving InvokeWith).return this->InvokeWith(ArgumentTuple());}
};

? ? ? ? 該模板類定義的模板類型就是函數的返回值類型——R。比如例子中Online方法,它被mock之后,傳導到該類的R就是bool。上面代碼中05行使用返回類型重新定義了函數類型為F()。06行別名了用于保存函數參數的元組類型為ArgumentTuple。08行定義的With函數是用于對參數的篩選。于是我們是以無參數函數為例,所以該處沒有設定參數預期。12行是我們mock函數的真實實現。這些內容我們將在之后詳細講解,我們再回到GMOCK_METHOD0_的定義上

 GMOCK_RESULT_(tn, __VA_ARGS__) ct Method( \) constness { \GTEST_COMPILE_ASSERT_((::testing::tuple_size<                          \tn ::testing::internal::Function<__VA_ARGS__>::ArgumentTuple>::value \== 0), \this_method_does_not_take_0_arguments); \GMOCK_MOCKER_(0, constness, Method).SetOwnerAndName(this, #Method); \return GMOCK_MOCKER_(0, constness, Method).Invoke(); \} \

? ? ? ??GMOCK_RESULT_宏定義了mock函數的返回類型

#define GMOCK_RESULT_(tn, ...) \tn ::testing::internal::Function<__VA_ARGS__>::Resulttemplate <typename R>
struct Function<R()> {typedef R Result;typedef ::testing::tuple<> ArgumentTuple;typedef typename MatcherTuple<ArgumentTuple>::type ArgumentMatcherTuple;typedef void MakeResultVoid();typedef IgnoredValue MakeResultIgnoredValue();
};

? ? ? ? 這么定義的一個優點就是可以通過模板將函數類型的定義中的返回類型給拆出來。這和FunctionMocker定義方式是一樣的。
? ? ? ??GTEST_COMPILE_ASSERT_宏用于檢測定義的參數個數是否符合規定。檢測完之后,使用FunctionMocker模板類對象的SetOwnerAndName方法將對象指針和方法名傳遞到底層邏輯中。最后就會調用FunctionMocker模板類對象的Invoke方法實現函數行為邏輯的調用。

? ? ? ??GMOCK_METHOD0_中還定義了另一個方法

  ::testing::MockSpec<__VA_ARGS__>& \gmock_##Method() constness { \GMOCK_MOCKER_(0, constness, Method).RegisterOwner(this); \return GMOCK_MOCKER_(0, constness, Method).With(); \} \

? ? ? ? 它使用了gmock和函數名組合成為一個新的函數。該函數內部通過FunctionMocker模板類對象的RegisterOwner方法保存了對象指針,最后返回了MockSpec模板對象。MockSpec模板對象在之前我們見過,它是為了實現參數篩選而設計的。其具體實現我們在之后會分析。

? ? ? ? 無參數的版本忽略了很多函數參數的問題,但是其讓我們可以清晰的看見實現的脈絡。現在我們將以有兩個參數的版本來講解其實現。

// INTERNAL IMPLEMENTATION - DON'T USE IN USER CODE!!!
#define GMOCK_METHOD2_(tn, constness, ct, Method, ...) \GMOCK_RESULT_(tn, __VA_ARGS__) ct Method( \GMOCK_ARG_(tn, 1, __VA_ARGS__) gmock_a1, \GMOCK_ARG_(tn, 2, __VA_ARGS__) gmock_a2) constness { \GTEST_COMPILE_ASSERT_((::testing::tuple_size<                          \tn ::testing::internal::Function<__VA_ARGS__>::ArgumentTuple>::value \== 2), \this_method_does_not_take_2_arguments); \GMOCK_MOCKER_(2, constness, Method).SetOwnerAndName(this, #Method); \return GMOCK_MOCKER_(2, constness, Method).Invoke(gmock_a1, gmock_a2); \} \::testing::MockSpec<__VA_ARGS__>& \gmock_##Method(GMOCK_MATCHER_(tn, 1, __VA_ARGS__) gmock_a1, \GMOCK_MATCHER_(tn, 2, __VA_ARGS__) gmock_a2) constness { \GMOCK_MOCKER_(2, constness, Method).RegisterOwner(this); \return GMOCK_MOCKER_(2, constness, Method).With(gmock_a1, gmock_a2); \} \mutable ::testing::FunctionMocker<__VA_ARGS__> GMOCK_MOCKER_(2, constness, \Method)

? ? ? ? 上例中,我們發現相關函數的定義多了兩個參數聲明。我們先看和mock函數同名的函數的參數定義,它使用了GMOCK_ARG_宏指定參數類型

#define GMOCK_ARG_(tn, N, ...) \tn ::testing::internal::Function<__VA_ARGS__>::Argument##N

? ? ? ? Function模板類在之前我們反復見過,它的一個非常大的作用就是從函數類型中拆分出函數返回值類型和各個參數類型。因為之前以無參數函數為例,所以我們并沒有欣賞到它的妙處。

template <typename R, typename A1>
struct Function<R(A1)>: Function<R()> {typedef A1 Argument1;typedef ::testing::tuple<A1> ArgumentTuple;typedef typename MatcherTuple<ArgumentTuple>::type ArgumentMatcherTuple;typedef void MakeResultVoid(A1);typedef IgnoredValue MakeResultIgnoredValue(A1);
};template <typename R, typename A1, typename A2>
struct Function<R(A1, A2)>: Function<R(A1)> {typedef A2 Argument2;typedef ::testing::tuple<A1, A2> ArgumentTuple;typedef typename MatcherTuple<ArgumentTuple>::type ArgumentMatcherTuple;typedef void MakeResultVoid(A1, A2);typedef IgnoredValue MakeResultIgnoredValue(A1, A2);
};

? ? ? ? Function模板類的兩個參數的版本繼承于一個參數的版本,一個參數版本繼承于無參數版本。這樣兩個參數版本中,它從無參數版本中繼承到了

typedef R Result;

? ? ? ? 從一個參數版本中繼承到了

typedef A1 Argument1;

? ? ? ? 而自身定義了

typedef A2 Argument2;

? ? ? ? 它還覆蓋了基類中ArgumentTuple、ArgumentMatcherTuple等的定義。

? ? ? ? 我們看到兩個參數版本的Function類的模板類型是R(A1, A2),這種方式就是函數類型的定義。而R、A1和A2是Function模板類的模板。以Login方法為例

MOCK_METHOD2(Login, bool(const std::string&, const std::string&));

? ? ? ? 編譯器將推導出R是bool,A1和A2都是const ?std::string&。這樣它便將函數返回類型和參數進行了拆分。并別名了各個類型,從而方便在之后模板中忽略具體類型。
? ? ? ?相應的FunctionMocker也是使用相同的方式實現了拆分,我們看下兩個參數版本的實現

template <typename R, typename A1, typename A2>
class FunctionMocker<R(A1, A2)> : publicinternal::FunctionMockerBase<R(A1, A2)> {public:typedef R F(A1, A2);typedef typename internal::Function<F>::ArgumentTuple ArgumentTuple;MockSpec<F>& With(const Matcher<A1>& m1, const Matcher<A2>& m2) {this->current_spec().SetMatchers(::testing::make_tuple(m1, m2));return this->current_spec();}R Invoke(A1 a1, A2 a2) {// Even though gcc and MSVC don't enforce it, 'this->' is required// by the C++ standard [14.6.4] here, as the base class type is// dependent on the template argument (and thus shouldn't be// looked into when resolving InvokeWith).return this->InvokeWith(ArgumentTuple(a1, a2));}
};

? ? ? ? 相比于無參數版本,它在With函數中使用了SetMatchers方法實現了參數限制,并在Invoke中,使用兩個參數定義了一個臨時的參數元組類型ArgumentTuple對象。這樣將參數放到一個元組對象中,是對InvokeWith方法對不同個數、不同類型、不同順序的參數調用實現統一化處理。

EXPECT_CALL、ON_CALL宏? ? ? ??

? ? ? ? 在介紹MOCK_METHOD系列宏是,我們發現其在我們mock的類中定義兩個方法和一個變量:

  1. GMOCK_RESULT_(tn, __VA_ARGS__) ct Method(……)
  2. ::testing::MockSpec<__VA_ARGS__>& gmock_##Method(……)
  3. mutable ::testing::FunctionMocker<__VA_ARGS__> GMOCK_MOCKER_(0, constness, Method)

? ? ? ? 1中的方法和我們希望mock的方法同名,這將方便使用者調用它。2中的函數是使用gmock和函數名聯合組裝成的新函數名,它返回了一個被參數篩選的函數對象。EXPECT_CALL和ON_CALL宏中就是調用了它。

#define GMOCK_ON_CALL_IMPL_(obj, call) \((obj).gmock_##call).InternalDefaultActionSetAt(__FILE__, __LINE__, \#obj, #call)
#define ON_CALL(obj, call) GMOCK_ON_CALL_IMPL_(obj, call)#define GMOCK_EXPECT_CALL_IMPL_(obj, call) \((obj).gmock_##call).InternalExpectedAt(__FILE__, __LINE__, #obj, #call)
#define EXPECT_CALL(obj, call) GMOCK_EXPECT_CALL_IMPL_(obj, call)

? ? ? ? 宏中((obj).gmock_##call)就是調用了2中的方法,并對返回的對象調用InternalDefaultActionSetAt或InternalExpectedAt。以下面的調用為例

EXPECT_CALL(test_user, Pay(_)).WillRepeatedly(testing::Return(true));

? ? ? ? 其最終是這樣的調用

test_user.gmock_Pay(_).InternalExpectedAt(__FILE__, __LINE__, 'test_user', 'Pay').WillRepeatedly(testing::Return(true));

? ? ? ? 下劃線_是通配符,它的定義如下

const internal::AnythingMatcher _ = {};class AnythingMatcher {public:template <typename T>operator Matcher<T>() const { return A<T>(); }
};template <typename T>
inline Matcher<T> A() { return MakeMatcher(new internal::AnyMatcherImpl<T>()); }template <typename T>
inline Matcher<T> MakeMatcher(const MatcherInterface<T>* impl) {return Matcher<T>(impl);
}

? ? ? ? 為什么任何函數的參數都可以接受AnythingMatcher。我們可以見2中參數的定義

  ::testing::MockSpec<__VA_ARGS__>& \gmock_##Method(GMOCK_MATCHER_(tn, 1, __VA_ARGS__) gmock_a1) constness { \GMOCK_MOCKER_(1, constness, Method).RegisterOwner(this); \return GMOCK_MOCKER_(1, constness, Method).With(gmock_a1); \} \// The matcher type for argument N of the given function type.// INTERNAL IMPLEMENTATION - DON'T USE IN USER CODE!!!#define GMOCK_MATCHER_(tn, N, ...) \const ::testing::Matcher<GMOCK_ARG_(tn, N, __VA_ARGS__)>&

? ? ? ? 函數的參數類型都是Matcher模板類,而AnythingMatcher定義了Matcher<T>()方法用于返回一個Matcher<T>對象。

參數過濾 ? ? ? ?

? ? ? ?參數過濾是Gmock非常有用的一個功能,它讓我們可以通過參數定義不同的調用場景。

? ? ? ?Gmock中提供了兩處設置參數過濾的地方,舉個例子

EXPECT_CALL(test_user, Pay(Eq(1))).With(_).WillRepeatedly(testing::Return(true));

? ? ? ?Pay中指定參數不能等于1,With則表示對參數沒有限制。這就是兩處參數約束。一般來說gmock##Method中的參數約束是針對各自參數的,而With則是關注于參數之間的關系。我們看下這兩處約束是怎么工作的。
? ? ? ? 以一個參數的版本為例,MOCK_METHOD1宏中

  ::testing::MockSpec<__VA_ARGS__>& \gmock_##Method(GMOCK_MATCHER_(tn, 1, __VA_ARGS__) gmock_a1) constness { \GMOCK_MOCKER_(1, constness, Method).RegisterOwner(this); \return GMOCK_MOCKER_(1, constness, Method).With(gmock_a1); \} \mutable ::testing::FunctionMocker<__VA_ARGS__> GMOCK_MOCKER_(1, constness, \Method)

? ? ? ? gmock##Method方法調用了FunctionMocker模板類的With方法,該方法返回了一個MockSpec模板對象。gmock##Method方法是在EXPECT_CALL宏中被調用的。
? ? ? ??FunctionMocker中的With是這么實現的

  MockSpec<F>& With(const Matcher<A1>& m1) {this->current_spec().SetMatchers(::testing::make_tuple(m1));return this->current_spec();}

? ? ? ??current_spec()是其基類FunctionMockerBase的方法,它返回了以FunctionMocker<R(A1)> 為模板的MockSpec對象。SetMatchers方法將參數的匹配規則設置到其底層的matchers_中

  void SetMatchers(const ArgumentMatcherTuple& matchers) {matchers_ = matchers;}

? ? ? ??matchers_這個匹配規則會在調用EXPECT_CALL時通過下列兩個方法保存起來

  // Adds a new default action spec to the function mocker and returns// the newly created spec.internal::OnCallSpec<F>& InternalDefaultActionSetAt(const char* file, int line, const char* obj, const char* call) {LogWithLocation(internal::kInfo, file, line,string("ON_CALL(") + obj + ", " + call + ") invoked");return function_mocker_->AddNewOnCallSpec(file, line, matchers_);}// Adds a new expectation spec to the function mocker and returns// the newly created spec.internal::TypedExpectation<F>& InternalExpectedAt(const char* file, int line, const char* obj, const char* call) {const string source_text(string("EXPECT_CALL(") + obj + ", " + call + ")");LogWithLocation(internal::kInfo, file, line, source_text + " invoked");return function_mocker_->AddNewExpectation(file, line, source_text, matchers_);}

? ? ? ? 具體的保存邏輯是

  // Adds and returns a default action spec for this mock function.OnCallSpec<F>& AddNewOnCallSpec(const char* file, int line,const ArgumentMatcherTuple& m)GTEST_LOCK_EXCLUDED_(g_gmock_mutex) {Mock::RegisterUseByOnCallOrExpectCall(MockObject(), file, line);OnCallSpec<F>* const on_call_spec = new OnCallSpec<F>(file, line, m);untyped_on_call_specs_.push_back(on_call_spec);return *on_call_spec;}// Adds and returns an expectation spec for this mock function.TypedExpectation<F>& AddNewExpectation(const char* file,int line,const string& source_text,const ArgumentMatcherTuple& m)GTEST_LOCK_EXCLUDED_(g_gmock_mutex) {Mock::RegisterUseByOnCallOrExpectCall(MockObject(), file, line);TypedExpectation<F>* const expectation =new TypedExpectation<F>(this, file, line, source_text, m);const linked_ptr<ExpectationBase> untyped_expectation(expectation);untyped_expectations_.push_back(untyped_expectation);// Adds this expectation into the implicit sequence if there is one.Sequence* const implicit_sequence = g_gmock_implicit_sequence.get();if (implicit_sequence != NULL) {implicit_sequence->AddExpectation(Expectation(untyped_expectation));}return *expectation;}

? ? ? ? 即以該規則為參數,新建了OnCallSpec<F>或TypedExpectation<F>對象,這兩個對象將會被保存到各自的vector中。當mock的函數被調用時,Gmock將通過下面兩個函數之一去檢測參數是否匹配

  // Returns the ON_CALL spec that matches this mock function with the// given arguments; returns NULL if no matching ON_CALL is found.// L = *const OnCallSpec<F>* FindOnCallSpec(const ArgumentTuple& args) const {for (UntypedOnCallSpecs::const_reverse_iterator it= untyped_on_call_specs_.rbegin();it != untyped_on_call_specs_.rend(); ++it) {const OnCallSpec<F>* spec = static_cast<const OnCallSpec<F>*>(*it);if (spec->Matches(args))return spec;}return NULL;}
  // Returns the expectation that matches the arguments, or NULL if no// expectation matches them.TypedExpectation<F>* FindMatchingExpectationLocked(const ArgumentTuple& args) constGTEST_EXCLUSIVE_LOCK_REQUIRED_(g_gmock_mutex) {g_gmock_mutex.AssertHeld();for (typename UntypedExpectations::const_reverse_iterator it =untyped_expectations_.rbegin();it != untyped_expectations_.rend(); ++it) {TypedExpectation<F>* const exp =static_cast<TypedExpectation<F>*>(it->get());if (exp->ShouldHandleArguments(args)) {return exp;}}return NULL;}

? ? ? ? 這兩個函數最終將在Matches函數中進行參數匹配

  // Returns true iff this expectation matches the given arguments.bool Matches(const ArgumentTuple& args) constGTEST_EXCLUSIVE_LOCK_REQUIRED_(g_gmock_mutex) {g_gmock_mutex.AssertHeld();return TupleMatches(matchers_, args) && extra_matcher_.Matches(args);}

? ? ? ? 這個函數中,還有extra_matcher_這種參數匹配規則。它是通過TypedExpectation模板類的With方法(不是FunctionMocker模板類的With方法)傳遞進來的

  // Implements the .With() clause.TypedExpectation& With(const Matcher<const ArgumentTuple&>& m) {if (last_clause_ == kWith) {ExpectSpecProperty(false,".With() cannot appear ""more than once in an EXPECT_CALL().");} else {ExpectSpecProperty(last_clause_ < kWith,".With() must be the first ""clause in an EXPECT_CALL().");}last_clause_ = kWith;extra_matcher_ = m;extra_matcher_specified_ = true;return *this;}

? ? ? ? 總結一下,Gmock的參數匹配通過FunctionMocker的With方法設置了一個通用匹配規則,還可以通過TypedExpectation的With方法設置額外的匹配規則,只有這兩個匹配規則都滿足時,才會被選中。

設定約束

? ? ? ? 我們主要分析下Times、WillOnce和WillRepeatedly這幾個常見的約束。先回顧一個例子

EXPECT_CALL(test_user, Pay(_)).Times(5).WillOnce(testing::Return(true)).WillOnce(testing::Return(true)).WillRepeatedly(testing::Return(false));

? ? ? ? 這例子說,Pay行為有5次可控的執行次數,第6次執行就按默認值返回了。第1個WillOnce規定第一次執行Pay的行為,第2個WillOnce規定第二次執行Pay的行為,之后的3~5次都按WillRepeatedly規定的方式去執行。

? ? ? ? 我們先看Times的實現

  // Implements the .Times() clause.TypedExpectation& Times(const Cardinality& a_cardinality) {ExpectationBase::UntypedTimes(a_cardinality);return *this;}// Implements the .Times() clause.TypedExpectation& Times(int n) {return Times(Exactly(n));}
// Implements the .Times() clause.
void ExpectationBase::UntypedTimes(const Cardinality& a_cardinality) {if (last_clause_ == kTimes) {ExpectSpecProperty(false,".Times() cannot appear ""more than once in an EXPECT_CALL().");} else {ExpectSpecProperty(last_clause_ < kTimes,".Times() cannot appear after "".InSequence(), .WillOnce(), .WillRepeatedly(), ""or .RetiresOnSaturation().");}last_clause_ = kTimes;SpecifyCardinality(a_cardinality);
}// Explicitly specifies the cardinality of this expectation.  Used by
// the subclasses to implement the .Times() clause.
void ExpectationBase::SpecifyCardinality(const Cardinality& a_cardinality) {cardinality_specified_ = true;cardinality_ = a_cardinality;
}

? ? ? ? 執行次數最終被轉換為Cardinality類的一個對象保存在FunctionMocker模板對象中。它將在IsSatisfied、IsSaturated和IsOverSaturated方法中被使用,用以判定執行的次數是否符合約定

  // Returns true iff this expectation is satisfied.bool IsSatisfied() constGTEST_EXCLUSIVE_LOCK_REQUIRED_(g_gmock_mutex) {g_gmock_mutex.AssertHeld();return cardinality().IsSatisfiedByCallCount(call_count_);}// Returns true iff this expectation is saturated.bool IsSaturated() constGTEST_EXCLUSIVE_LOCK_REQUIRED_(g_gmock_mutex) {g_gmock_mutex.AssertHeld();return cardinality().IsSaturatedByCallCount(call_count_);}// Returns true iff this expectation is over-saturated.bool IsOverSaturated() constGTEST_EXCLUSIVE_LOCK_REQUIRED_(g_gmock_mutex) {g_gmock_mutex.AssertHeld();return cardinality().IsOverSaturatedByCallCount(call_count_);}

? ? ? ? 參數中的call_count_就是函數執行的次數,它是在IncrementCallCount函數中實現自增。IncrementCallCount函數則是在獲取行為時被調用到

  const Action<F>* GetActionForArguments(const FunctionMockerBase<F>* mocker,const ArgumentTuple& args,::std::ostream* what,::std::ostream* why)GTEST_EXCLUSIVE_LOCK_REQUIRED_(g_gmock_mutex) {g_gmock_mutex.AssertHeld();if (IsSaturated()) {
......IncrementCallCount();
......}IncrementCallCount();
......}

? ? ? ? 我們再看下WillOnce的實現

  // Implements the .WillOnce() clause.TypedExpectation& WillOnce(const Action<F>& action) {ExpectSpecProperty(last_clause_ <= kWillOnce,".WillOnce() cannot appear after "".WillRepeatedly() or .RetiresOnSaturation().");last_clause_ = kWillOnce;untyped_actions_.push_back(new Action<F>(action));if (!cardinality_specified()) {set_cardinality(Exactly(static_cast<int>(untyped_actions_.size())));}return *this;}

? ? ? ? 可見WillOnce將入參重新賦值給一個新建的Action<F>對象。然后將它保存到untyped_actions_列表中。最終它會在GetCurrentAction方法中,通過參數匹配后被取出

    return count <= action_count ?*static_cast<const Action<F>*>(untyped_actions_[count - 1]) :repeated_action();

? ? ? ? 上面代碼中的repeated_action方法是在WillRepeatedly方法這被賦值的

  // Implements the .WillRepeatedly() clause.TypedExpectation& WillRepeatedly(const Action<F>& action) {if (last_clause_ == kWillRepeatedly) {ExpectSpecProperty(false,".WillRepeatedly() cannot appear ""more than once in an EXPECT_CALL().");} else {ExpectSpecProperty(last_clause_ < kWillRepeatedly,".WillRepeatedly() cannot appear ""after .RetiresOnSaturation().");}last_clause_ = kWillRepeatedly;repeated_action_specified_ = true;repeated_action_ = action;if (!cardinality_specified()) {set_cardinality(AtLeast(static_cast<int>(untyped_actions_.size())));}// Now that no more action clauses can be specified, we check// whether their count makes sense.CheckActionCountIfNotDone();return *this;}

? ? ? ? 再看下testing::Return(true)是怎么轉換為Action<F>的。它的定義是

template <typename R>
internal::ReturnAction<R> Return(R value) {return internal::ReturnAction<R>(internal::move(value));
}

? ? ? ? 其中ReturnAction是個模板類,它重載了Action<F>()方法,將返回值轉換為一個Action<F>對象

  template <typename F>operator Action<F>() const {typedef typename Function<F>::Result Result;GTEST_COMPILE_ASSERT_(!is_reference<Result>::value,use_ReturnRef_instead_of_Return_to_return_a_reference);return Action<F>(new Impl<R, F>(value_));}

? ? ? ? 在new一個Action<F>是,傳入了一個Impl模板類對象,這個模板類有一個Perform方法,其實現就是返回期待的值

virtual Result Perform(const ArgumentTuple&) { return value_; }

? ? ? ? 那么Action<F>對象和這個Impl模板類是怎么聯系的呢?我們看下Impl的定義

  template <typename R_, typename F>class Impl : public ActionInterface<F> {public:
……

? ? ? ? 而在Action模板類的內部有

template <typename F>
class Action {
……
Action(const Action& action) : impl_(action.impl_) {}
……Result Perform(const ArgumentTuple& args) const {
……    return impl_->Perform(args);}private:
……internal::linked_ptr<ActionInterface<F> > impl_;
};

? ? ? ? 很醒目,最終執行的行為將由Action類中的Impl_成員變量來執行,而該Impl_變量就是在Action被創建時傳入的。

執行

? ? ? ? 當我們調用mock的類的mock的函數時,將會調用到MOCK_METHOD系列宏中定義的函數。以一個參數版本為例

  GMOCK_RESULT_(tn, __VA_ARGS__) ct Method( \GMOCK_ARG_(tn, 1, __VA_ARGS__) gmock_a1) constness { \GTEST_COMPILE_ASSERT_((::testing::tuple_size<                          \tn ::testing::internal::Function<__VA_ARGS__>::ArgumentTuple>::value \== 1), \this_method_does_not_take_1_argument); \GMOCK_MOCKER_(1, constness, Method).SetOwnerAndName(this, #Method); \return GMOCK_MOCKER_(1, constness, Method).Invoke(gmock_a1); \} \

? ? ? ? 其最終調用到FunctionMocker類的Invoke函數中

  R Invoke(A1 a1) {// Even though gcc and MSVC don't enforce it, 'this->' is required// by the C++ standard [14.6.4] here, as the base class type is// dependent on the template argument (and thus shouldn't be// looked into when resolving InvokeWith).return this->InvokeWith(ArgumentTuple(a1));}

? ? ? ? 調用InvokeWith之前,已將參數轉換成一個ArgumentTuple對象,這樣將方便之后統一處理。? ? ? ??InvokeWith函數內部使用了一個結果承載器——ResultHolder用于保存結果。InvokeWith最終會調用到FunctionMockerBase的PerformDefaultAction中

  Result PerformDefaultAction(const ArgumentTuple& args,const string& call_description) const {const OnCallSpec<F>* const spec =this->FindOnCallSpec(args);if (spec != NULL) {return spec->GetAction().Perform(args);}const string message = call_description +"\n    The mock function has no default action ""set, and its return type has no default value set.";
#if GTEST_HAS_EXCEPTIONSif (!DefaultValue<Result>::Exists()) {throw std::runtime_error(message);}
#elseAssert(DefaultValue<Result>::Exists(), "", -1, message);
#endifreturn DefaultValue<Result>::Get();}

? ? ? ? 第3行通過參數匹配相應的處理行為。找到行為后,在06行執行該行為;沒有找到,則返回默認值。

? ? ? ? 至此,Gmock的主要流程相關的源碼已經分析結束了。我們稍微總結下:

  • Mock的類通過MOCK_METHOD系列方法,聲明了一個Mock函數的對象,并定義了一個通過該對象獲取符合相應約束的函數對象。還定義了一個和需要mock的函數同名的函數,該函數內部完成最終的結果計算。
  • EXPECT_CALL宏和WillOnce、WillRepeatedly等方法,設定了函數對象的一些特性。
  • 最終用戶調用函數時,將通過參數匹配得到適合的函數對象,并執行該函數對象中的預期行為。



總結

以上是生活随笔為你收集整理的Google Mock(Gmock)简单使用和源码分析——源码分析的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

日韩视频一区二区在线观看 | 免费在线观看日韩欧美 | 欧美激情综合五月色丁香小说 | 久久免费国产 | 精选久久| 亚洲国产视频直播 | 久草电影在线观看 | 久久精品久久久精品美女 | 香蕉网在线观看 | 成人在线小视频 | 免费中文字幕 | 欧洲视频一区 | 夜色资源站wwwcom | 亚洲最大av网 | 91网站免费观看 | 在线观看免费一区 | 激情网色 | 九九热re | 在线免费黄色 | 精品国产乱码一区二区三区在线 | 超碰97网站 | 国产男女爽爽爽免费视频 | 亚洲成aⅴ人片久久青草影院 | 欧美激情奇米色 | 少妇bbb搡bbbb搡bbbb | 国产尤物一区二区三区 | 99视频在线免费播放 | 久久久久99精品成人片三人毛片 | 337p日本欧洲亚洲大胆裸体艺术 | 婷婷亚洲五月 | 免费69视频 | 中文字幕成人在线 | 色中文字幕在线观看 | 一本一道久久a久久综合蜜桃 | 久久91久久久久麻豆精品 | 在线草 | 99精品福利 | 丁五月婷婷 | 久久久久久久网站 | 日本中文不卡 | 99精品福利 | 色婷婷视频在线 | 免费特级黄毛片 | 草久久久| 色综合久久综合中文综合网 | 中文字幕在线看 | 综合色站导航 | 国产另类av | 免费亚洲成人 | 在线色吧 | 久久久久久久久久免费 | 午夜影院日本 | 四虎成人精品永久免费av | 丁香六月婷婷开心 | 欧美日韩首页 | 国产精品一区二区三区观看 | 国产精品白浆视频 | 米奇影视7777| 在线 你懂| 久久综合偷偷噜噜噜色 | 91精品入口 | 久久99精品国产91久久来源 | 国产在线观看网站 | 国产精品热视频 | 国产一二三精品 | 91丝袜美腿 | 欧美久久久影院 | 国产 在线观看 | 成人污视频在线观看 | 亚洲一区精品二人人爽久久 | 欧美日韩精品在线免费观看 | a极黄色片 | 国产成人福利在线观看 | 天天干天天射天天插 | 国产拍揄自揄精品视频麻豆 | 亚洲国产一区二区精品专区 | 久久精品一区二区三区中文字幕 | 亚洲精品免费在线 | 波多野结衣电影久久 | 91丨九色丨蝌蚪丨对白 | 天天操天天谢 | 国产亚洲成av人片在线观看桃 | 天天干天天操av | 欧美在线99 | 在线午夜电影神马影院 | 亚洲精品合集 | 久久深夜| 久久久久久久久久电影 | 日韩电影久久久 | 日韩免费看的电影 | 97人人射| 精品一区二区三区香蕉蜜桃 | 狠狠干干| 国偷自产视频一区二区久 | 天天干天天草天天爽 | 国产黄网站在线观看 | 日韩免费区 | 992tv在线观看网站 | 三级黄色在线 | 日韩三级一区 | 一区二区精品在线 | 97人人网 | 久久伊人八月婷婷综合激情 | 黄网站app在线观看免费视频 | 国产在线免费观看 | 国产精品久久久久久久久久久久 | 97精品在线观看 | 亚洲成色 | 夜夜高潮夜夜爽国产伦精品 | 天天爽天天射 | 黄av免费在线观看 | 亚洲国产999 | 精品国产乱码久久久久久1区二区 | 黄色电影在线免费观看 | 精品国产精品国产偷麻豆 | 99久久精品国产一区二区三区 | 日韩在线观看的 | 久久久精品视频网站 | 又污又黄的网站 | 国产99久久九九精品免费 | 中文字幕在线观看三区 | 免费久久99精品国产婷婷六月 | 色婷av | 久久精品视频在线观看免费 | 五月丁色 | 久久免费电影网 | 成年人视频在线观看免费 | www一起操 | 黄色日本片| 日韩免费一区二区 | 国产精品99久久久久久久久 | 久久国产精品小视频 | 国产精品久久久久久久久久免费 | 国产精品 中文字幕 亚洲 欧美 | 成人av在线电影 | 一区在线播放 | 中文字幕精品久久 | 亚洲免费高清视频 | 欧洲一区精品 | 国产成人精品午夜在线播放 | 天堂av色婷婷一区二区三区 | 国产精品久久久久久久久久久杏吧 | 欧美日韩视频一区二区 | 96av麻豆蜜桃一区二区 | 国产精品a级 | 中文字幕一区二区三 | 久久96国产精品久久99漫画 | 国产又粗又猛又黄视频 | 欧美日韩在线观看视频 | 色停停五月天 | 久久综合给合久久狠狠色 | 最近免费观看的电影完整版 | 在线 国产 亚洲 欧美 | 天天干天天想 | 2021国产精品视频 | 中文免费观看 | 免费看一级片 | 精品视频在线观看 | 日韩精品一区二区在线观看 | 国产黄a三级 | 在线视频欧美精品 | 国产精品黄色 | 亚洲精品免费观看视频 | 亚洲欧美日韩在线一区二区 | 一级精品视频在线观看宜春院 | 国产一级片免费视频 | aaaaaa毛片 | 成人18视频 | 免费观看一区二区三区视频 | 国产一区二区三区免费在线 | 黄色福利网站 | 久久午夜羞羞影院 | 国产精品美女999 | 欧美在线视频免费 | 激情欧美日韩一区二区 | 特黄特色特刺激视频免费播放 | 国产免费av一区二区三区 | 亚洲人在线7777777精品 | 久久人人爽人人人人片 | 2022久久国产露脸精品国产 | 日韩国产欧美在线视频 | 97在线观看免费高清完整版在线观看 | 国产精品一区二区久久精品爱涩 | 久久久麻豆精品一区二区 | 俺要去色综合狠狠 | 免费久久片| av成人免费在线观看 | 国产精品久久久久久久99 | 欧美日韩亚洲精品在线 | 狠狠的干狠狠的操 | 国产精品久久久久久久久免费看 | 免费观看av网站 | 久久久久成人精品免费播放动漫 | 手机看片午夜 | 日韩最新理论电影 | 国产精品成人在线观看 | 国内精品久久久久久久97牛牛 | 91精品区| 日韩精品短视频 | 亚洲国产精品成人精品 | 在线黄色av电影 | 手机av在线免费观看 | 黄色一级大片免费看 | 成人在线视频免费看 | 久草在线免费看视频 | 国产午夜精品一区二区三区四区 | 日韩精品一区二区三区视频播放 | 中文字幕在线观看第三页 | 中文字幕免费播放 | 欧美在线视频a | 国内精品视频免费 | 日韩影视大全 | 伊人干综合 | 激情网站 | 黄色免费国产 | 国产高清免费在线观看 | 国产欧美综合视频 | 亚洲精品乱码久久久久v最新版 | 国产人成精品一区二区三 | 久久综合干 | 五月婷婷亚洲 | 久久欧洲视频 | www.成人精品 | 四虎影视成人 | 国产在线精品一区二区 | 麻豆国产精品视频 | av蜜桃在线 | www欧美日韩| 在线观看mv的中文字幕网站 | 国产主播99 | 国产精品大片免费观看 | 久久免费视频3 | 亚洲美女免费视频 | 日本精品免费看 | 中文字幕91视频 | 996久久国产精品线观看 | 五月天婷婷免费视频 | 精品国产一区二区三区久久影院 | 亚洲欧美日韩精品一区二区 | 青青草国产精品 | 91精品人成在线观看 | 97在线视频网站 | 91九色综合 | 久久高清免费观看 | 亚洲视频电影在线 | 亚洲一区精品人人爽人人躁 | 亚洲高清精品在线 | 欧美精品乱码久久久久久 | 91日韩精品一区 | 亚洲精品乱码久久 | 日韩在线视频免费播放 | 成年人免费观看在线视频 | 亚洲情感电影大片 | 国产成人在线播放 | 午夜日b视频 | 免费看黄网站在线 | 狠狠狠干 | 91原创在线观看 | 欧美精品久久久久 | 国产亚洲免费观看 | 国产精品av免费 | 91高清完整版在线观看 | 有码中文在线 | 国产手机视频 | 国产精品aⅴ| 最新av免费| 成 人 黄 色 视频播放1 | 91成人网在线观看 | 国产原厂视频在线观看 | 日韩精品第一区 | 亚洲1区 在线 | 97人人艹 | 六月丁香激情综合色啪小说 | 欧美精品一区二区免费 | 亚洲国产精品电影 | 国产99亚洲 | 婷婷综合激情 | 2019天天干天天色 | 视频99爱 | 免费看色的网站 | 亚洲精品国偷拍自产在线观看 | 人人看人人艹 | 亚洲精品视频免费在线观看 | 高清av免费观看 | 草久久久久久 | 狠狠色丁香婷婷综合 | 五月婷久久 | 欧美日韩亚洲精品在线 | 久久艹艹 | 91av在线视频免费观看 | 久久99国产精品久久99 | 在线成人欧美 | 免费97视频 | 欧美巨大荫蒂茸毛毛人妖 | 在线观看欧美成人 | 欧产日产国产69 | 国产淫片免费看 | 久久国产精品99久久久久久老狼 | 91在线你懂的 | 久久久久二区 | 天天爱天天操 | 69国产精品视频免费观看 | 欧美性超爽 | 久综合网 | 精品国内自产拍在线观看视频 | 国产午夜精品一区二区三区欧美 | 人人玩人人弄 | 99精品免费久久久久久日本 | www.色午夜.com | 精品毛片一区二区免费看 | 亚洲最新合集 | 久久久久久久久久免费视频 | 在线国产视频 | 国产区免费| 国产精品一区二区三区免费看 | 日日干干夜夜 | 99视频网站 | 国产精品麻豆视频 | 日日操网 | 亚洲欧洲一区二区在线观看 | 亚洲电影av在线 | 欧美亚洲三级 | 国产精品一区欧美 | 99热精品国产| 免费a网| 久草a视频| 亚洲视频,欧洲视频 | 日韩精品一区二区三区在线视频 | 国产91av视频在线观看 | av在线收看| 免费试看一区 | 国产精品久久视频 | 国产97碰免费视频 | 在线免费观看麻豆视频 | 国产午夜精品一区二区三区欧美 | 久久免费av电影 | 超碰人人91 | 综合成人在线 | 中文字幕视频在线播放 | 在线免费观看麻豆视频 | 精品一区 精品二区 | 亚洲综合成人av | 激情文学综合丁香 | 免费在线观看污网站 | 国产精品久久久久久久av电影 | 日韩精品视频在线观看网址 | 免费aa大片| 奇米影视999 | 日韩毛片一区 | 超级碰99| 国产中文a | 深夜国产在线 | 欧美xxxxx在线视频 | 国产亚洲一区二区在线观看 | av在线电影免费观看 | 九九久久婷婷 | 开心综合网 | 免费在线观看av片 | 福利视频一二区 | 日韩视频免费观看高清完整版在线 | 中文十次啦 | 夜夜骑天天操 | 免费高清在线观看成人 | 免费在线观看中文字幕 | 中国精品一区二区 | 又黄又爽又无遮挡的视频 | 狠狠躁18三区二区一区ai明星 | 欧美一级日韩免费不卡 | 久久精品五月 | 99热精品久久 | 久久精品毛片 | 一区二区三区免费在线 | 成人免费在线观看电影 | 亚洲永久精品一区 | 中文字幕精品www乱入免费视频 | 丁香五月网久久综合 | 国产高清黄色 | 又黄又刺激视频 | 97超碰人人模人人人爽人人爱 | 久久综合久色欧美综合狠狠 | 久久精品99国产精品酒店日本 | 亚洲91中文字幕无线码三区 | 99精品国产兔费观看久久99 | 天天弄天天干 | 精品国产一区二区三区久久久蜜月 | 亚洲综合色激情五月 | 天天射天天爽 | 久久成人国产 | 亚洲精选在线观看 | 色偷偷88888欧美精品久久久 | 在线观看久久 | 欧美日韩精 | 婷婷丁香导航 | 久久久人人爽 | 成人高清av在线 | 国产高清视频在线观看 | 日韩精品中文字幕av | 欧美热久久 | 日韩电影在线观看一区二区 | 亚洲成人中文在线 | 日韩欧美专区 | 国产一区在线免费观看视频 | 久久狠狠一本精品综合网 | 97爱 | 午夜精品一区二区三区四区 | 日韩欧美一区二区在线观看 | 怡红院av久久久久久久 | 亚洲激情在线播放 | 国产一级二级在线 | 一区在线观看 | 香蕉网站在线观看 | 国产高清在线永久 | 免费看黄在线 | 在线观看中文字幕2021 | 女人18毛片a级毛片一区二区 | 久久国产亚洲精品 | 中文字幕在线人 | av在观看 | 久久99国产视频 | 免费观看久久久 | av大片免费在线观看 | 日韩av中文 | 韩国精品一区二区三区六区色诱 | 亚洲 欧洲 国产 日本 综合 | 成年人视频在线免费 | 夜夜高潮夜夜爽国产伦精品 | 五月天婷亚洲天综合网鲁鲁鲁 | 国产传媒一区在线 | 久久久久久久综合色一本 | 久久精品视频2 | 韩国av一区 | 狠狠躁夜夜躁人人爽视频 | 日韩av快播电影网 | 插插插色综合 | www.天天综合| 日韩中文字幕视频在线观看 | 高清av网站 | 国产首页 | 久久久久久久久久久成人 | 91精品少妇偷拍99 | 免费看污污视频的网站 | 亚洲精品在线网站 | v片在线播放 | 国产成人a亚洲精品 | 2024av| 午夜视频播放 | 亚洲日b视频 | 91成人精品国产刺激国语对白 | 黄色视屏免费在线观看 | 午夜av电影院| 91九色精品国产 | 一级一级一片免费 | 日本中文字幕在线一区 | 久久久午夜剧场 | 欧美日韩一区二区在线观看 | 国产字幕在线观看 | 曰韩精品 | 狠狠的操狠狠的干 | 六月丁香婷 | 久久99国产精品二区护士 | 国产免费成人 | 麻豆视频在线观看免费 | 成人精品国产 | 免费看三片 | 黄色综合 | 久久国产视屏 | 久久久国产精品人人片99精片欧美一 | 福利一区二区 | 亚洲精品在线资源 | 日韩欧三级| 成人在线视频你懂的 | 夜色在线资源 | 综合久久婷婷 | 99久久99久国产黄毛片 | 国产一区二区三区高清播放 | 欧美日韩久久不卡 | 国产青草视频在线观看 | 欧美日韩二区在线 | 中文字幕乱码在线播放 | 婷婷丁香花五月天 | 国内精品久久久久久中文字幕 | 中文字幕色播 | 国产一区二区视频在线播放 | 日韩免| 国产精品欧美久久 | 久久超碰在线 | 国产一区二区三区在线 | 国产精品久久久免费 | 久久久久国产一区二区三区四区 | av在线免费网站 | 一区二区伦理电影 | 欧美日韩国语 | 韩国av在线播放 | 天天爱天天操 | 激情五月婷婷综合网 | 国产一区二区高清视频 | 亚洲人成免费 | 一区二区高清在线 | 亚州精品视频 | 国产一级高清视频 | 欧美片一区二区三区 | 看黄色91| 午夜私人影院 | 色婷婷激情五月 | 日韩欧美电影在线 | 色网站国产精品 | www.国产高清 | 美女久久久久久 | 国产精品a久久 | 五月天色婷婷丁香 | 国产精品免费观看网站 | 伊人欧美 | 人人舔人人舔 | 人人草人人草 | 久草在线在线视频 | 亚洲五月花 | 麻豆视频在线看 | 黄色免费观看网址 | 成人h电影 | 97理论电影 | 午夜av日韩 | 精品国产综合区久久久久久 | 久久国产露脸精品国产 | 人人澡人人爽 | 在线免费观看av网站 | 中文国产字幕 | 国产视频在线播放 | 在线国产小视频 | 国产精品久久久久久久久久ktv | 国产成人无码AⅤ片在线观 日韩av不卡在线 | 日韩av图片| 日韩欧美国产成人 | 国产视频一区在线免费观看 | 亚洲精品乱码久久久久久久久久 | 中文字幕在线播放一区二区 | 在线观看一级 | 久久亚洲综合国产精品99麻豆的功能介绍 | www五月天com | 黄色1级大片 | 在线看黄网站 | 免费观看性生活大片 | 久久激情五月婷婷 | 精品久久久久国产免费第一页 | 中文字幕视频三区 | 久久国产亚洲精品 | 91色在线观看视频 | 特级西西www44高清大胆图片 | 欧美aⅴ在线观看 | 五月天,com | 啪嗒啪嗒免费观看完整版 | www.五月天色 | 国产黄色免费电影 | 免费观看91视频大全 | 天堂va欧美va亚洲va老司机 | 综合久久婷婷 | 欧美最猛性xxxxx(亚洲精品) | 日日草夜夜操 | 欧美亚洲精品在线观看 | 在线免费高清视频 | 久久久在线| 精品久久久久一区二区国产 | 免费观看完整版无人区 | 久久男人免费视频 | 97国产电影 | 成人在线一区二区三区 | 久久欧美在线电影 | 久久久www免费电影网 | 国产国语在线 | 欧美成人在线网站 | 国产小视频在线看 | 久久久九色精品国产一区二区三区 | 99视频+国产日韩欧美 | 91精品国产乱码久久 | 欧美一区二区精美视频 | 婷五月天激情 | 色诱亚洲精品久久久久久 | 精品久久久网 | 香蕉精品视频在线观看 | 成人全视频免费观看在线看 | 久久成年视频 | 欧美污污网站 | 成人黄色大片在线免费观看 | 国产精品久久久久久久久软件 | 麻花传媒mv免费观看 | 欧洲精品亚洲精品 | 久草在线视频中文 | 久久久久女人精品毛片 | 国产精品一区二区三区久久 | 天天操福利视频 | 亚洲精品资源在线 | 亚洲男男gaygay无套 | 欧美精彩视频在线观看 | 久草视频在线免费播放 | 欧美一区二区三区在线 | av不卡中文字幕 | 久久久久久蜜av免费网站 | 国产高清视频在线 | 亚洲视频精品 | 日韩在线一二三区 | 亚洲国产成人在线播放 | 深夜精品福利 | 91麻豆精品国产91久久久使用方法 | 在线久草视频 | 亚洲国产一二三 | 久久精品网站免费观看 | 成年人在线看片 | 亚洲国产成人久久综合 | 黄色看片 | 99热在| 日韩在线不卡视频 | 久久另类小说 | 久久激情视频网 | 九九热只有这里有精品 | 在线观影网站 | 久久精品二区 | av千婊在线免费观看 | 人人爽人人爽人人爽 | www.色午夜.com| 日本久久久久久久久久 | 日韩欧美在线观看一区二区 | 亚洲视频1| 亚洲精品午夜aaa久久久 | 超碰国产在线观看 | 综合久久久久久久 | 99精品国产视频 | 亚洲激情久久 | 97av视频在线观看 | 97在线影视 | 国产精品二区在线观看 | 99久久综合国产精品二区 | 黄色网址中文字幕 | 九九久久精品 | 免费在线色电影 | 超碰在线网 | 日韩综合视频在线观看 | 96看片| 有码视频在线观看 | 这里只有精品视频在线观看 | 欧美日韩在线视频一区 | 久久优 | 成人网中文字幕 | 一级c片| 在线久久 | 亚洲国产一二三 | 久久综合桃花 | 久久综合之合合综合久久 | 亚洲三区在线 | 91秒拍国产福利一区 | a视频在线播放 | 久久伊人婷婷 | 深夜免费小视频 | 亚洲欧洲一区二区在线观看 | 国产在线黄色 | 97超碰网| 91在线你懂的 | 中文字幕在线免费看线人 | 99久久这里有精品 | 亚洲永久精品国产 | 欧美在线不卡一区 | 五月天久久久 | 激情网在线观看 | 国产日韩欧美自拍 | 综合网婷婷 | 日日干日日 | 日日干干夜夜 | avlulu久久精品 | 四虎www. | 成人av网站在线观看 | 免费看黄在线 | 日韩精品一区二区三区在线视频 | av+在线播放在线播放 | 99九九99九九九视频精品 | 色婷五月 | 欧美福利网站 | 九九久久影视 | 日韩欧美xxxx | 欧美一区二区三区在线 | 久久人人97超碰精品888 | 日韩国产精品久久久久久亚洲 | 精品一区91 | 精品国产一区二区三区av性色 | 中文字幕在线免费观看视频 | 国产精品中文字幕av | 精品中文字幕在线播放 | 精品欧美一区二区在线观看 | 伊人干综合 | 日本色小说视频 | 99久久久国产精品美女 | 免费网站在线观看成人 | 黄色成人小视频 | 四虎成人网 | 欧美成人精品三级在线观看播放 | 激情图片qvod | 91精品国产麻豆国产自产影视 | 黄色在线免费观看网址 | 精品少妇一区二区三区在线 | 一区二区三区在线观看免费 | 久久国产美女视频 | 91色吧 | 一区二区三区精品在线 | 女人高潮特级毛片 | 国产麻豆果冻传媒在线观看 | 一区二区三区在线不卡 | 奇米影视四色8888 | 久久久久综合精品福利啪啪 | 国产亚洲高清视频 | 免费日韩高清 | 欧美日韩高清一区 | 午夜视频一区二区三区 | 不卡av电影在线观看 | 欧美a级一区二区 | 久久首页| 免费久久久久久 | 黄色av网站在线观看 | 国产美女无遮挡永久免费 | av成人在线观看 | 天天五月天色 | 在线观看深夜视频 | 九九热只有这里有精品 | 国内精品久久久久影院一蜜桃 | 久久久影院一区二区三区 | 97精品国产91久久久久久 | 亚洲视频久久久 | 91精品在线观看视频 | 一区 在线观看 | 欧美精品在线免费 | 欧美最猛性xxxxx亚洲精品 | 日韩在线观看av | 91九色性视频 | 日本xxxx.com | 日本一区二区三区免费观看 | 欧美视频18 | 黄a在线观看 | 女人高潮特级毛片 | 久久不射网站 | 亚洲亚洲精品在线观看 | 91精品国产麻豆国产自产影视 | 国产精品色视频 | 在线播放 一区 | 精品视频在线免费观看 | 色wwwww| 国产一级免费视频 | 国产精品一区二区免费视频 | 亚洲国产精品女人久久久 | 日韩免费一级a毛片在线播放一级 | 成人a v视频 | 久久国产精品久久久久 | 中文字幕免费 | 欧美激情另类 | 亚洲精品视频免费 | 一级片观看| 亚洲欧美怡红院 | 国产精品美女在线 | 97视频免费播放 | 手机在线中文字幕 | 国产高清无线码2021 | 中文字幕日韩一区二区三区不卡 | 麻豆视频在线免费 | 国产手机在线播放 | 99国产精品一区 | 欧美国产一区二区 | 久久精品看片 | 国产91精品看黄网站在线观看动漫 | 在线播放精品一区二区三区 | 国产成人av福利 | 天天操夜夜逼 | 国产经典 欧美精品 | 成人av在线一区二区 | 天天操天天操天天操天天操 | 中国美女一级看片 | 国产伦精品一区二区三区照片91 | 天天操夜夜叫 | 国产麻豆精品在线观看 | 久久综合久久久久88 | 国产精品久久久久久久久久久久久久 | 豆豆色资源网xfplay | 日本在线成人 | 91在线免费观看网站 | 特级黄色电影 | 国产精品va在线播放 | 国产成人精品一区二 | 久久人网 | 国产精品久久久久久一区二区三区 | 狠狠躁天天躁综合网 | 亚洲国内精品在线 | 亚洲高清视频在线观看免费 | 黄色毛片一级 | 狠狠狠综合| 国产这里只有精品 | 五月天激情综合 | 99中文字幕视频 | 久久久九色精品国产一区二区三区 | 久久久久亚洲最大xxxx | 最近中文字幕国语免费av | 久久经典国产视频 | 一区二区三区免费看 | 中文字幕一区二区三区久久 | 五月激情婷婷丁香 | 狠狠干在线 | 日韩中出在线 | 日本成人黄色片 | 伊人久久在线观看 | 日韩精品一区二区在线 | 久久国产精品99精国产 | 在线精品视频免费播放 | 久久久久黄色 | 亚洲精品乱码久久久久久蜜桃不爽 | 最近免费中文字幕 | 午夜免费福利视频 | 久久综合之合合综合久久 | 黄网站免费大全入口 | 日韩精品一区不卡 | www在线观看国产 | 欧美日韩在线免费观看视频 | av不卡中文字幕 | 亚洲色图 校园春色 | 国产精品一区二区免费 | 国内精品久久久久久久 | 国产精品一区二区在线 | 国产精品原创av片国产免费 | 欧美久久久久久久久久久久 | 2022久久国产露脸精品国产 | 国产精品久久久久久欧美 | 国产一区播放 | 久久久久9999亚洲精品 | 九九视频一区 | 国产精品免费视频一区二区 | 国产在线视频一区二区 | 日本在线中文在线 | 久久,天天综合 | 国产 一区二区三区 在线 | 亚洲观看黄色网 | 欧美久久久久 | 91精品国产综合久久久久久久 | 日日日网| 国产精品原创在线 | 国内外激情视频 | 成人免费观看大片 | 国产在线观看你懂得 | www视频免费在线观看 | 99久久婷婷国产综合亚洲 | 四虎成人精品永久免费av九九 | 亚洲精品系列 | 99精品国产免费久久久久久下载 | 国产精品久久久久久久久久东京 | 国产精品一区二区久久精品爱微奶 | 国产精品第二页 | 91黄色小网站 | 亚洲精品视频免费在线观看 | 91精品国产一区 | 成年人网站免费在线观看 | 久久人人97超碰com | 日韩精品久久久 | 免费在线观看av网址 | 久久久亚洲麻豆日韩精品一区三区 | 日韩免费三级 | 韩国在线一区 | 欧美日韩国语 | 欧美激情在线看 | 天天综合婷婷 | 国产二区免费视频 | www.五月天婷婷 | 8x成人在线 | 亚洲欧美日本国产 | 日韩欧美99 | 亚洲精品高清视频 | 伊人天堂网 | 免费黄色网址网站 | 成年人黄色在线观看 | 亚洲aⅴ在线 | 不卡av电影在线观看 | 国产伦理久久精品久久久久_ | 91视频在线看 | www.色婷婷 | 国产精品久久久久国产a级 激情综合中文娱乐网 | 日韩av免费观看网站 | 久草在线欧美 | 国内小视频在线观看 | 久草在线精品观看 | 一区二区三区四区五区六区 | 99久热在线精品视频观看 | 久久伊人色综合 | 日韩在线观看一区二区三区 | 日韩一区二区久久 | 中文在线中文资源 | 久久婷婷影视 | 亚洲伦理精品 | 色夜视频| 夜夜躁日日躁 | www.av在线.com| 在线黄av| 中文av资源站 | 97超碰在线资源 | 米奇影视7777| 婷婷丁香七月 | 欧美久草在线 | 在线观看www.| 欧美动漫一区二区三区 | 中文字幕一区二区三区在线播放 | 91污在线| 国产h在线播放 | 欧美日韩在线精品 | 欧美一级高清片 | 天天操天天射天天操 | 五月激情站 | 久久99视频精品 | 日韩在线视频观看 | 中文字幕91 | 国产又粗又猛又黄又爽视频 | 亚洲最大在线视频 | 精品国产一区二区三区在线 | 亚洲理论视频 | 99精品免费久久久久久久久 | 日韩一级理论片 | 91亚洲在线 | 欧美日韩一区二区三区不卡 | 狠狠色丁香久久婷婷综合_中 | 亚洲理论在线 | 视频在线观看亚洲 | 中文字幕视频网 | 婷婷色综合色 | 亚洲精品在线观看中文字幕 | 久久国色夜色精品国产 | 天天操综合网 | 国产精品久久久久久久电影 | 国产精品午夜免费福利视频 | 成人91免费视频 | 国产亚洲va综合人人澡精品 | 亚洲观看黄色网 | 久久综合偷偷噜噜噜色 | 在线观看国产成人av片 | 91在线网址 | 99精品视频99 | 操久久免费视频 | 亚洲va欧洲va国产va不卡 | 97在线观看 | 日韩av手机在线看 | 公开超碰在线 | 黄色片视频免费 | 国产91区 | 天天翘av| 亚洲黄网址| 中文字幕精品三级久久久 | 久草精品视频在线观看 | 国产一区二区久久 | 国产理论片在线观看 | 国产福利在线 | 免费黄a大片| 中文区中文字幕免费看 | 亚洲视频在线播放 | 在线中文日韩 | 国产三级精品在线 | 国产亚洲欧美日韩高清 | 97免费在线观看视频 | www婷婷 | 欧美性生爱 | 婷婷久久久久 | 日韩精品一区不卡 | 日本精品va在线观看 | 黄色av一级 | 亚洲精品一区二区三区新线路 | 国产精品黄色 | 黄色亚洲片 | 美女黄濒| 不卡精品视频 | av天天澡天天爽天天av | 成人免费看视频 | 91精品久久久久久粉嫩 | 正在播放久久 | 色婷婷综合久色 | 色婷婷88av视频一二三区 | 国产码电影 | 81精品国产乱码久久久久久 | av中文国产| 伊人久久电影网 | 99视频国产精品 | 国产精品18p | 色综合久久久久综合99 | 国产护士hd高朝护士1 | 日韩av专区 | 婷婷在线五月 | 欧日韩在线 | 日本黄色免费看 | 欧美一区二区在线免费观看 | 在线免费av电影 | 天天拍天天色 | 亚洲免费在线观看视频 | 中文字幕在线观看第三页 | av在线播放观看 | 亚洲精品一区二区网址 | 91精品伦理 | 色偷偷男人的天堂av | 96看片 | 中文字幕亚洲综合久久五月天色无吗'' | 久草在线视频在线 | 国产99久久久精品视频 | 九色91视频 | 国产在线观看污片 |