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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

cks32和stm32_cks子,间谍,局部Mo子和短管

發布時間:2023/12/3 编程问答 50 豆豆
生活随笔 收集整理的這篇文章主要介紹了 cks32和stm32_cks子,间谍,局部Mo子和短管 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

cks32和stm32

本文是我們名為“ 用Mockito測試 ”的學院課程的一部分。

在本課程中,您將深入了解Mockito的魔力。 您將了解有關“模擬”,“間諜”和“部分模擬”的信息,以及它們相應的存根行為。 您還將看到使用測試雙打和對象匹配器進行驗證的過程。 最后,討論了使用Mockito的測試驅動開發(TDD),以了解該庫如何適合TDD的概念。 在這里查看 !

目錄

1.簡介 2.模擬,存根,間諜–名稱是什么? 3.存根方法 4.存根返回值
4.1。 使用答案 4.2。 有關行為驅動開發測試約定的說明 4.3。 在Eclipse中使用Mockito靜態方法的提示 4.4。 使用多個模擬 4.5。 測試自己! 測試更新!
5.參數匹配器 6.間諜和部分存根 7.結論 8.下載源代碼

1.簡介

在本教程中,我們將深入研究使用Mockito存根類和接口。

2.模擬,存根,間諜–名稱是什么?

嘲笑中的許多術語可以互換使用,也可以作為動詞和名詞使用。 我們現在將對這些術語進行定義,以避免將來造成混淆。

  • 模擬(名詞) –一個對象,充當另一個對象的雙精度對象。
  • 模擬(動詞) –創建模擬對象或對方法進行存根。
  • 間諜(名詞) –裝飾現有對象并允許對該對象的方法進行存根和對該對象的調用進行驗證的對象。
  • 間諜(動詞) –創建和使用間諜對象。
  • 存根(名詞) –可以在調用方法時提供“罐頭答案”的對象。
  • 存根(動詞) –創建固定答案。
  • Partial Mock,Partial Stub(動詞) –間諜的另一種術語,其中某些方法已被禁用。

從技術上講,Mockito是一個測試間諜框架,而不是模擬框架,因為它使我們能夠創建間諜和驗證行為,以及創建具有存根行為的模擬對象。

正如在上一教程中所看到的,我們可以使用when().thenReturn()方法對給定接口或類的行為進行存根。 現在,我們將研究為Mocks和Spies提供存根的所有方法。

3.存根方法

給定以下界面:

public interface Printer {void printTestPage();}

以下是使用它的基于字符串緩沖區的簡單化“文字處理器”類:

public class StringProcessor {private Printer printer;private String currentBuffer;public StringProcessor(Printer printer) {this.printer = printer;}public Optional<String> statusAndTest() {printer.printTestPage();return Optional.ofNullable(currentBuffer);}}

我們要編寫一個測試方法,該方法將測試構造后是否缺少當前緩沖區并處理測試頁的打印。

這是我們的測試課程:

public class StringProcessorTest {private Printer printer;@Testpublic void internal_buffer_should_be_absent_after_construction() {// GivenStringProcessor processor = new StringProcessor(printer);// WhenOptional<String> actualBuffer = processor.statusAndTest();// ThenassertFalse(actualBuffer.isPresent());} }

我們知道statusAndTest()將涉及對Printer的printTestPage()方法的調用,并且printer引用未初始化,因此如果執行此測試,我們將以NullPointerException結尾。 為了避免這種情況,我們只需要注釋測試類以告訴JUnit使用Mockito運行它,并注釋Printer作為一個模擬,以告訴mockito為此創建一個模擬。

@RunWith(MockitoJUnitRunner.class) public class StringProcessorTest {@Mockprivate Printer printer;@Testpublic void internal_buffer_should_be_absent_after_construction() {// GivenStringProcessor processor = new StringProcessor(printer);// WhenOptional<String> actualBuffer = processor.statusAndTest();// ThenassertFalse(actualBuffer.isPresent());}}

現在我們可以執行測試,Mockito將為我們創建Printer的實現,并將其實例分配給printer變量。 我們將不再獲得NullPointerException。

但是,如果Printer是一類實際完成某些工作的類,例如打印物理測試頁,該怎么辦? 如果我們選擇了@Spy而不是創建@Mock怎么辦? 記住,除非被偵聽,否則間諜會在類上調用間諜的真實方法。 我們希望避免在調用該方法時做任何實際的事情。 讓我們做一個簡單的Printer實現:

public class SysoutPrinter implements Printer {@Overridepublic void printTestPage() {System.out.println("This is a test page");}}

并將其作為間諜添加到我們的測試類中,并添加一個新方法來測試使用它:

@Spyprivate SysoutPrinter sysoutPrinter;@Testpublic void internal_buffer_should_be_absent_after_construction_sysout() {// GivenStringProcessor processor = new StringProcessor(sysoutPrinter);// WhenOptional<String> actualBuffer = processor.statusAndTest();// ThenassertFalse(actualBuffer.isPresent());}

如果現在執行此測試,您將在控制臺上看到以下輸出:

This is a test page

這確認我們的測試用例實際上是在執行SysoutPrinter類的真實方法,這是因為它是Spy而不是Mock。 如果該類實際執行了測試頁的實際物理打印,那將是非常不希望的!

當我們執行部分模擬或Spy時,可以使用org.mockito.Mockito.doNothing()調用的方法進行存根,以確保其中沒有任何org.mockito.Mockito.doNothing() 。

讓我們添加以下導入和測試:

import static org.mockito.Mockito.*;@Testpublic void internal_buffer_should_be_absent_after_construction_sysout_with_donothing() {// GivenStringProcessor processor = new StringProcessor(sysoutPrinter);doNothing().when(sysoutPrinter).printTestPage();// WhenOptional<String> actualBuffer = processor.statusAndTest();// ThenassertFalse(actualBuffer.isPresent());}

注意方法doNothing.when(sysoutPrinter).printTestPage() :這告訴Mockito當調用@Spy sysoutPrinter的void方法printTestPage ,不應執行真正的方法,而應執行任何操作。 現在,當我們執行此測試時,屏幕上看不到任何輸出。

如果未連接物理打印機,如果我們擴展打印機接口以引發新的PrinterNotConnectedException異常,該怎么辦? 我們如何測試這種情況?

首先,讓我們創建一個非常簡單的新異常類。

public class PrinterNotConnectedException extends Exception {private static final long serialVersionUID = -6643301294924639178L;}

并修改我們的界面以將其拋出:

void printTestPage() throws PrinterNotConnectedException;

如果拋出異常,我們還需要修改StringProcessor以執行某些操作。 為了簡單起見,我們只將異常拋出給調用類。

public Optional<String> statusAndTest() throws PrinterNotConnectedException

現在我們要測試異常是否傳遞給調用類,因此我們必須強制打印機拋出異常。 與doNothing()類似,我們可以使用doThrow強制執行異常。

讓我們添加以下測試:

@Test(expected = PrinterNotConnectedException.class)public void printer_not_connected_exception_should_be_thrown_up_the_stack() throws Exception {// GivenStringProcessor processor = new StringProcessor(printer);doThrow(new PrinterNotConnectedException()).when(printer).printTestPage();// WhenOptional<String> actualBuffer = processor.statusAndTest();// ThenassertFalse(actualBuffer.isPresent());}

在這里,我們看到可以使用doThrow()拋出所需的任何異常。 在這種情況下,我們將拋出滿足我們測試要求的PrinterNotConnectedException 。

現在我們已經學習了如何對void方法進行存根,讓我們看一下返回一些數據。

4.存根返回值

讓我們開始創建一個數據訪問對象,以從數據庫中持久化和檢索客戶對象。 該DAO將使用內部的企業java EntityManager接口進行實際的數據庫交互。

為了使用EntityManager我們將使用JPA 2.0的Hibernate實現,將以下依賴項添加到pom.xml中:

<dependency><groupId>org.hibernate.javax.persistence</groupId><artifactId>hibernate-jpa-2.0-api</artifactId><version>1.0.1.Final</version></dependency>

現在,我們將創建一個簡單的Customer實體來表示要保留的Customer。

@Entity public class Customer {@Id @GeneratedValueprivate long id;private String name;private String address;public Customer() {}public Customer(long id, String name, String address) {super();this.id = id;this.name = name;this.address = address;}public long getId() {return id;}public void setId(long id) {this.id = id;}public String getAddress() {return address;}public void setAddress(String address) {this.address = address;}public String getName() {return name;}public void setName(String name) {this.name = name;}}

現在,我們將創建一個骨架DAO,該骨架使用@PersistenceContext配置注入的EntityManager 。 我們不必擔心使用Java持久性體系結構(JPA)或它如何工作-我們將使用Mockito完全繞過它,但這是Mockito實際應用的一個很好的實例。

public class CustomerDAO {@PersistenceContextEntityManager em;public CustomerDAO(EntityManager em) {this.em = em;}}

我們將在DAO中添加基本的“檢索和更新”功能,并使用Mockito對其進行測試。

首先使用Retrieve方法-我們將傳遞一個ID并從數據庫中返回適當的Customer(如果存在)。

public Optional<Customer> findById(long id) throws Exception {return Optional.ofNullable(em.find(Customer.class, id));}

在這里,我們使用Java Optional來避免對結果進行空檢查。

現在,我們可以添加測試以在找到客戶但找不到客戶的地方測試此方法–我們將使用Mockito方法org.mockito.Mockito.when對find()方法進行存根處理以在每種情況下返回適當的Optional。然后thenReturn()

讓我們如下創建Test類(為Mockito方法import static org.mockito.Mockito.*; ):

@RunWith(MockitoJUnitRunner.class) public class CustomerDAOTest {private CustomerDAO dao;@Mockprivate EntityManager mockEntityManager;@Beforepublic void setUp() throws Exception {dao = new CustomerDAO(mockEntityManager);}@Testpublic void finding_existing_customer_should_return_customer() throws Exception {// Givenlong expectedId = 10;String expectedName = "John Doe";String expectedAddress = "21 Main Street";Customer expectedCustomer = new Customer(expectedId, expectedName, expectedAddress);when(mockEntityManager.find(Customer.class, expectedId)).thenReturn(expectedCustomer);// WhenOptional<Customer> actualCustomer = dao.findById(expectedId);// ThenassertTrue(actualCustomer.isPresent());assertEquals(expectedId, actualCustomer.get().getId());assertEquals(expectedName, actualCustomer.get().getName());assertEquals(expectedAddress, actualCustomer.get().getAddress());} }

我們看到了用于啟用模仿,模仿EntityManger并將其注入到測試中的類的常用樣板。 讓我們看一下測試方法。

第一行涉及創建具有已知期望值的Customer ,然后我們看到對Mockito的調用告訴我們,當使用我們提供的特定輸入參數調用EntityManager.find()方法時,該客戶將返回此客戶。 然后,我們執行findById()方法和一組斷言的實際執行,以確保我們獲得了期望的值。

讓我們剖析Mockito調用:

when(mockEntityManager.find(Customer.class, expectedId)).thenReturn(expectedCustomer);

這演示了Mockito強大而優雅的語法。 讀起來幾乎像普通的英語。 當find()的方法mockEntityManager對象被稱為與特定輸入Customer.class和expectedId ,然后返回expectedCustomer對象。

如果您使用未告知其期望的參數調用Mock,則它將僅返回null,如以下測試所示:

@Testpublic void invoking_mock_with_unexpected_argument_returns_null() throws Exception {// Givenlong expectedId = 10L;long unexpectedId = 20L;String expectedName = "John Doe";String expectedAddress = "21 Main Street";Customer expectedCustomer = new Customer(expectedId, expectedName, expectedAddress);when(mockEntityManager.find(Customer.class, expectedId)).thenReturn(expectedCustomer);// WhenOptional<Customer> actualCustomer = dao.findById(unexpectedId);// ThenassertFalse(actualCustomer.isPresent());}

您還可以在不同的時間對Mock進行存根,以實現不同的行為,具體取決于輸入。 讓我們讓Mock根據輸入的ID返回其他客戶:

@Testpublic void invoking_mock_with_different_argument_returns_different_customers() throws Exception {// Givenlong expectedId1 = 10L;String expectedName1 = "John Doe";String expectedAddress1 = "21 Main Street";Customer expectedCustomer1 = new Customer(expectedId1, expectedName1, expectedAddress1);long expectedId2 = 20L;String expectedName2 = "Jane Deer";String expectedAddress2 = "46 High Street";Customer expectedCustomer2 = new Customer(expectedId2, expectedName2, expectedAddress2);when(mockEntityManager.find(Customer.class, expectedId1)).thenReturn(expectedCustomer1);when(mockEntityManager.find(Customer.class, expectedId2)).thenReturn(expectedCustomer2);// WhenOptional<Customer> actualCustomer1 = dao.findById(expectedId1);Optional<Customer> actualCustomer2 = dao.findById(expectedId2);// ThenassertEquals(expectedName1, actualCustomer1.get().getName());assertEquals(expectedName2, actualCustomer2.get().getName());}

您甚至可以鏈接返回,以使模擬在每次調用時執行不同的操作。 請注意,如果您調用模擬程序的次數超過了您的存根行為,那么它將永遠永遠根據最后一個存根行為。

@Testpublic void invoking_mock_with_chained_stubs_returns_different_customers() throws Exception {// Givenlong expectedId1 = 10L;String expectedName1 = "John Doe";String expectedAddress1 = "21 Main Street";Customer expectedCustomer1 = new Customer(expectedId1, expectedName1, expectedAddress1);long expectedId2 = 20L;String expectedName2 = "Jane Deer";String expectedAddress2 = "46 High Street";Customer expectedCustomer2 = new Customer(expectedId2, expectedName2, expectedAddress2);when(mockEntityManager.find(Customer.class, expectedId1)).thenReturn(expectedCustomer1).thenReturn(expectedCustomer2);// WhenOptional<Customer> actualCustomer1 = dao.findById(expectedId1);Optional<Customer> actualCustomer2 = dao.findById(expectedId1);// ThenassertEquals(expectedName1, actualCustomer1.get().getName());assertEquals(expectedName2, actualCustomer2.get().getName());}

請注意,我們輸入了相同的ID到兩個電話,不同的行為是由第二goverened theReturn()方法,這只能是因為when()存根的一部分明確預期和輸入expectedId1 ,如果我們通過expectedId2我們由于它不是存根中的期望值,因此會從模擬中獲得空響應。

現在讓我們測試一下缺少客戶的情況。

@Testpublic void finding_missing_customer_should_return_null() throws Exception {// Givenlong expectedId = 10L;when(mockEntityManager.find(Customer.class, expectedId)).thenReturn(null);// WhenOptional<Customer> actualCustomer = dao.findById(expectedId);// ThenassertFalse(actualCustomer.isPresent());}

在這里我們可以看到我們使用相同的語法,但是這次使用它來返回null。

允許的Mockito您使用的可變參數thenReturn存根連續調用,所以如果我們想我們可以在前面的兩個測試搟成一個如下:

@Testpublic void finding_customer_should_respond_appropriately() throws Exception {// Givenlong expectedId = 10L;String expectedName = "John Doe";String expectedAddress = "21 Main Street";Customer expectedCustomer1 = new Customer(expectedId, expectedName, expectedAddress);Customer expectedCustomer2 = null;when(mockEntityManager.find(Customer.class, expectedId)).thenReturn(expectedCustomer1, expectedCustomer2);// WhenOptional<Customer> actualCustomer1 = dao.findById(expectedId);Optional<Customer> actualCustomer2 = dao.findById(expectedId);// ThenassertTrue(actualCustomer1.isPresent());assertFalse(actualCustomer2.isPresent());}

如果我們的find方法由于某些持久性問題而引發異常怎么辦? 讓我們測試一下!

@Test(expected=IllegalArgumentException.class)public void finding_customer_should_throw_exception_up_the_stack() throws Exception {// Givenlong expectedId = 10L;when(mockEntityManager.find(Customer.class, expectedId)).thenThrow(new IllegalArgumentException());// Whendao.findById(expectedId);// Thenfail("Exception should be thrown.");}

我們使用了thenThrow()方法引發異常。 在對無效方法進行存根時,將此語法與我們對doThrow()使用進行doThrow() 。 這是兩個相似但不同的方法– thenThrow()將不適用于void方法。

使用答案

上面我們看到我們創建了一個具有某些期望值的客戶。 如果我們想創建一些已知的測試用戶并以id為基礎返回他們,則可以使用Answer ,可以從when()調用中返回。 Answer是Mockito提供的通用類型,用于提供“罐頭響應”。 它的answer()方法采用一個InvocationOnMock對象,該對象包含有關當前模擬方法調用的某些信息。

讓我們創建3個客戶和一個Answer,根據輸入的ID選擇要退回的客戶。

首先,將3個客戶添加為測試類的私有成員。

private Customer homerSimpson, bruceWayne, tyrionLannister;

然后添加一個專用的setupCustomers方法以對其進行初始化,然后從@Before方法進行調用。

@Beforepublic void setUp() throws Exception {dao = new CustomerDAO(mockEntityManager);setupCustomers();}private void setupCustomers() {homerSimpson = new Customer(1, "Homer Simpson", "Springfield");bruceWayne = new Customer(2, "Bruce Wayne", "Gotham City");tyrionLannister = new Customer(2, "Tyrion Lannister", "Kings Landing");}

現在,我們可以基于在運行時傳遞給傳遞給模擬EntityManager的find()方法的ID創建一個Answer來返回適當的Customer。

private Answer<Customer> withCustomerById = new Answer<Customer>() {@Overridepublic Customer answer(InvocationOnMock invocation) throws Throwable {Object[] args = invocation.getArguments();int id = ((Long)args[1]).intValue(); // Cast to int for switch.switch (id) {case 1 : return homerSimpson;case 2 : return bruceWayne;case 3 : return tyrionLannister;default : return null;}}};

我們可以看到我們使用InvocationOnMock提取了傳遞到Mock方法調用中的參數。 我們知道第二個參數是ID,因此我們可以讀取該參數并確定要返回的適當客戶。 稍后,帶有withCustomerById的答案的名稱將與我們的模擬語法匹配。

現在,讓我們編寫一個測試來證明此答案的實際效果。

@Testpublic void finding_customer_by_id_returns_appropriate_customer() throws Exception {// Givenlong[] expectedId = {1, 2, 3};when(mockEntityManager.find(eq(Customer.class), anyLong())).thenAnswer(withCustomerById);// WhenOptional<Customer> actualCustomer0 = dao.findById(expectedId[0]);Optional<Customer> actualCustomer1 = dao.findById(expectedId[1]);Optional<Customer> actualCustomer2 = dao.findById(expectedId[2]);// ThenassertEquals("Homer Simpson", actualCustomer0.get().getName());assertEquals("Bruce Wayne", actualCustomer1.get().getName());assertEquals("Tyrion Lannister", actualCustomer2.get().getName());}

讓我們詳細了解一下存根線。

when(mockEntityManager.find(eq(Customer.class), anyLong())).thenAnswer(withCustomerById);

在這里,我們看到了一些新事物。 第一件事是,我們不執行when().thenReturn()而是執行when().thenAnswer()并提供withCustomerById Answer作為要給出的答案。 第二件事是我們不對傳遞給mockEntityManager.find()的ID使用真實值,而是使用靜態的org.mockito.Matchers.anyLong() 。 這是一個Matcher ,用于使Mockito發出Answer,而無需檢查是否已傳入特定的Long值。Matchers讓我們忽略模擬調用的參數,而只專注于返回值。

我們還用eq() Matcher裝飾了Customer.class –這是由于您不能在Mock方法調用中混合使用實值和Matchers,您要么必須將所有參數都作為Matchers,要么必須將所有參數都作為實值。 eq()提供了一個Matcher,僅當運行時參數等于存根中的指定參數時才匹配。 讓我們繼續僅在輸入類類型為Customer.class類型時不指定特定ID的情況下才返回Answer。

什么這一切意味著,三個調用mockEntityManager.find()用不同的ID是所有產生相同的答案報錯,并且我們已經編碼的答案與不同的ID相應的客戶對象響應是我們已經成功地嘲笑一個EntityManager能力模仿現實行為。

有關行為驅動開發測試約定的說明

您可能已經注意到,我們在單元測試中采用了約定,將測試分為三部分– //給定,//時間和//然后。 該約定稱為行為驅動開發,是設計單元測試的一種非常合乎邏輯的方法。

  • // 給定的是設置階段,在該階段我們初始化數據和存根模擬類。 它與陳述“給定以下初始條件”相同。
  • //什么時候是執行階段,在該階段我們執行被測方法并捕獲所有返回的對象。
  • //然后是驗證階段,在此階段我們放置斷言邏輯,該邏輯將檢查該方法是否表現出預期的行為。

Mockito在org.mockito.BDDMockito類中開箱即用地支持BDD。 它用BDD doppelgangers – given() , willReturn() , willThrow() , willAnswer()替換了常規的存根方法– when() , thenReturn() , thenThrow() , thenAnswer()等。 這樣可以避免在// //給定部分中使用when() ,因為這可能會造成混淆。

因為我們在測試中使用BDD約定,所以我們還將使用BDDMockito提供的方法。

讓我們使用BDDMockito語法重寫finding_existing_customer_should_return_customer() 。

import static org.mockito.BDDMockito.*;@Testpublic void finding_existing_customer_should_return_customer_bdd() throws Exception {// Givenlong expectedId = 10L;String expectedName = "John Doe";String expectedAddress = "21 Main Street";Customer expectedCustomer = new Customer(expectedId, expectedName, expectedAddress);given(mockEntityManager.find(Customer.class, expectedId)).willReturn(expectedCustomer);// WhenOptional<Customer> actualCustomer = dao.findById(expectedId);// ThenassertTrue(actualCustomer.isPresent());assertEquals(expectedId, actualCustomer.get().getId());assertEquals(expectedName, actualCustomer.get().getName());assertEquals(expectedAddress, actualCustomer.get().getAddress());}

測試的邏輯沒有改變,只是以BDD格式可讀。

在Eclipse中使用Mockito靜態方法的提示

如果要避免導入org.mockito.Mockito.*等,為各種Mockito靜態方法手動添加靜態導入可能會很org.mockito.Mockito.*為了在Eclipse中為這些方法啟用內容輔助,您只需啟動org.mockito.Mockito.* > Preferences并轉到左側導航欄中的Java / Editor / Content Assist / Favorites。 然后,按照圖1添加以下內容作為“ New Type…”。

  • org.mockito.Mockito
  • org.mockito.Matchers
  • org.mockito.BDDMockito

這會將Mockito靜態方法添加到Eclipse Content Assist中,使您可以在使用它們時自動完成并導入它們。

圖1 – Content Assist收藏夾

使用多個模擬

現在,我們將結合在一起使用多個模擬。 讓我們向DAO中添加一個方法以返回所有可用客戶的列表。

public List<Customer> findAll() throws Exception {TypedQuery<Customer> query = em.createQuery("select * from CUSTOMER", Customer.class);return query.getResultList();}

在這里,我們看到EntityManager的createQuery()方法返回一個通用類型TypedQuery 。 它接受一個SQL String和一個作為返回類型的類作為參數。 TypedQuery本身公開了幾種方法,包括List getResultList() ,這些方法可用于執行返回多個值的查詢,例如上面的select * from CUSTOMER查詢中的select * from CUSTOMER 。

為了對此方法編寫測試,我們將要創建一個TypedQuery的Mock。

@Mock private TypedQuery<Customer> mockQuery;

現在,我們可以對這個模擬查詢進行存根以返回已知客戶的列表。 讓我們創建一個答案來做到這一點,并重用我們先前創建的已知客戶。 您可能已經注意到Answer是一個功能接口,只有一種方法。 我們正在使用Java 8,因此我們可以創建一個lambda表達式來表示我們的內聯Answer,而不是像前面的Answer示例中那樣創建一個匿名內部類。

given(mockQuery.getResultList()).willAnswer(i -> Arrays.asList(homerSimpson, bruceWayne, tyrionLannister));

當然我們也可以將上面的存根編碼為

given(mockQuery.getResultList()).willReturn(Arrays.asList(homerSimpson, bruceWayne, tyrionLannister));given

這展示了Mockito的靈活性–總是有幾種不同的方式來做相同的事情。

現在我們已經對模擬TypedQuery的行為進行了存根,我們可以對模擬EntityManager進行存根以在請求時返回它。 與其將SQL引入我們的測試用例中, anyString()僅使用anyString()匹配器來觸發模擬createQuery() ,當然,我們還將用eq()匹配器包圍該類參數。

完整的測試如下所示:

@Testpublic void finding_all_customers_should_return_all_customers() throws Exception {// Givengiven(mockQuery.getResultList()).willAnswer(i -> Arrays.asList(homerSimpson, bruceWayne, tyrionLannister));given(mockEntityManager.createQuery(anyString(), eq(Customer.class))).willReturn(mockQuery);// WhenList<Customer> actualCustomers = dao.findAll();// ThenassertEquals(actualCustomers.size(), 3);}

測試更新!

讓我們添加Update() DAO方法:

public Customer update(Customer customer) throws Exception {return em.merge(customer);}

現在查看是否可以為其創建測試。 本教程隨附的示例代碼項目中已編寫了可能的解決方案。 記住,在Mockito中有很多方法可以做相同的事情,看看是否能想到其中的幾種!

5.參數匹配器

Mocktio的自然行為是使用對象的equals()方法作為參數傳入,以查看是否存在特定的存根行為。 但是,如果對我們來說不重要的是那些值,則可以在存根時避免使用真實的對象和變量。 我們通過使用Mockito參數匹配器來實現

我們已經看到了一些運行中的Mockito參數匹配器: anyLong() , anyString()和eq 。 當我們不特別在意Mock的輸入時,我們會使用這些匹配器,我們只對編碼它的返回行為感興趣,并且我們希望它在所有條件下的行為都相同。

如前所述,但需要特別注意的是,當使用參數匹配器時,所有參數都必須是參數匹配器,您不能將實值與參數匹配器混合和匹配,否則會從Mockito中獲取運行時錯誤。

參數匹配器都擴展了org.mockito.ArgumentMatcher ,Mockito包含一個現成的參數匹配器庫,可以通過org.mockito.Matchers的靜態方法進行org.mockito.Matchers ,要使用它們只需導入org.mockito.Matchers.* ;

您可以查看org.mockito.Matchers的javadoc,以查看Mockito提供的所有Matchers,而以下測試類演示了其中一些用法:

package com.javacodegeeks.hughwphamill.mockito.stubbing;import static org.junit.Assert.*; import static org.mockito.Matchers.*; import static org.mockito.Mockito.*;import java.util.Arrays; import java.util.Collections; import java.util.List; import java.util.Map; import java.util.Set;import org.junit.Test; import org.junit.runner.RunWith; import org.mockito.Mock; import org.mockito.Mockito; import org.mockito.runners.MockitoJUnitRunner;@RunWith(MockitoJUnitRunner.class) public class MatchersTest {public interface TestForMock {public boolean usesPrimitives(int i, float f, double d, byte b, boolean bool);public boolean usesObjects(String s, Object o, Integer i);public boolean usesCollections(List<String> list, Map<Integer, String> map, Set<Object> set);public boolean usesString(String s);public boolean usesVarargs(String... s);public boolean usesObject(Object o);}@MockTestForMock test;@Testpublic void test() {// default behaviour is to return falseassertFalse(test.usesString("Hello"));when(test.usesObjects(any(), any(), any())).thenReturn(true);assertTrue(test.usesObjects("Hello", new Thread(), 17));Mockito.reset(test);when(test.usesObjects(anyString(), anyObject(), anyInt())).thenReturn(true);assertTrue(test.usesObjects("Hi there", new Float(18), 42));Mockito.reset(test);when(test.usesPrimitives(anyInt(), anyFloat(), anyDouble(), anyByte(), anyBoolean())).thenReturn(true);assertTrue(test.usesPrimitives(1, 43.4f, 3.141592654d, (byte)2, false));Mockito.reset(test);// Gives unchecked type conversion warningwhen(test.usesCollections(anyList(), anyMap(), anySet())).thenReturn(true);assertTrue(test.usesCollections(Arrays.asList("Hello", "World"), Collections.EMPTY_MAP, Collections.EMPTY_SET));Mockito.reset(test);// Gives no warningwhen(test.usesCollections(anyListOf(String.class), anyMapOf(Integer.class, String.class), anySetOf(Object.class))).thenReturn(true);assertTrue(test.usesCollections(Collections.emptyList(), Collections.emptyMap(), Collections.emptySet()));Mockito.reset(test);// eq() must match exactlywhen(test.usesObjects(eq("Hello World"), any(Object.class),anyInt())).thenReturn(true);assertFalse(test.usesObjects("Hi World", new Object(), 360));assertTrue(test.usesObjects("Hello World", new Object(), 360));Mockito.reset(test);when(test.usesString(startsWith("Hello"))).thenReturn(true);assertTrue(test.usesString("Hello there"));Mockito.reset(test);when(test.usesString(endsWith("something"))).thenReturn(true);assertTrue(test.usesString("isn't that something"));Mockito.reset(test);when(test.usesString(contains("second"))).thenReturn(true);assertTrue(test.usesString("first, second, third."));Mockito.reset(test);// Regular Expressionwhen(test.usesString(matches("^\\\\w+$"))).thenReturn(true);assertTrue(test.usesString("Weak_Password1"));assertFalse(test.usesString("@Str0nG!pa$$woR>%42"));Mockito.reset(test);when(test.usesString((String)isNull())).thenReturn(true);assertTrue(test.usesString(null));Mockito.reset(test);when(test.usesString((String)isNotNull())).thenReturn(true);assertTrue(test.usesString("Anything"));Mockito.reset(test);// Object ReferenceString string1 = new String("hello");String string2 = new String("hello");when(test.usesString(same(string1))).thenReturn(true);assertTrue(test.usesString(string1));assertFalse(test.usesString(string2));Mockito.reset(test);// Compare to eq()when(test.usesString(eq(string1))).thenReturn(true);assertTrue(test.usesString(string1));assertTrue(test.usesString(string2));Mockito.reset(test);when(test.usesVarargs(anyVararg())).thenReturn(true);assertTrue(test.usesVarargs("A","B","C","D","E"));assertTrue(test.usesVarargs("ABC", "123"));assertTrue(test.usesVarargs("Hello!"));Mockito.reset(test);when(test.usesObject(isA(String.class))).thenReturn(true);assertTrue(test.usesObject("A String Object"));assertFalse(test.usesObject(new Integer(7)));Mockito.reset(test);// Field equality using reflectionwhen(test.usesObject(refEq(new SomeBeanWithoutEquals("abc", 123)))).thenReturn(true);assertTrue(test.usesObject(new SomeBeanWithoutEquals("abc", 123)));Mockito.reset(test);// Compare to eq()when(test.usesObject(eq(new SomeBeanWithoutEquals("abc", 123)))).thenReturn(true);assertFalse(test.usesObject(new SomeBeanWithoutEquals("abc", 123)));Mockito.reset(test);when(test.usesObject(eq(new SomeBeanWithEquals("abc", 123)))).thenReturn(true);assertTrue(test.usesObject(new SomeBeanWithEquals("abc", 123)));Mockito.reset(test);}public class SomeBeanWithoutEquals {private String string;private int number;public SomeBeanWithoutEquals(String string, int number) {this.string = string;this.number = number;}}public class SomeBeanWithEquals {private String string;private int number;public SomeBeanWithEquals(String string, int number) {this.string = string;this.number = number;}@Overridepublic int hashCode() {final int prime = 31;int result = 1;result = prime * result + getOuterType().hashCode();result = prime * result + number;result = prime * result+ ((string == null) ? 0 : string.hashCode());return result;}@Overridepublic boolean equals(Object obj) {if (this == obj)return true;if (obj == null)return false;if (getClass() != obj.getClass())return false;SomeBeanWithEquals other = (SomeBeanWithEquals) obj;if (!getOuterType().equals(other.getOuterType()))return false;if (number != other.number)return false;if (string == null) {if (other.string != null)return false;} else if (!string.equals(other.string))return false;return true;}private MatchersTest getOuterType() {return MatchersTest.this;}} }

還可以通過擴展org.mockito.ArgumentMatcher來創建自己的org.mockito.ArgumentMatcher 。 讓我們創建一個匹配器,如果列表包含特定元素,該匹配器將觸發。 我們還將創建一個用于創建Matcher的靜態便利方法,該方法使用argThat將Matcher轉換為List,以便在存根調用中使用。 我們將實現matches()方法來調用List的contains方法來進行實際的contains檢查。

public class ListContainsMatcher<T> extends ArgumentMatcher<List<T>> {private T element;public ListContainsMatcher(T element) {this.element = element;}@Overridepublic boolean matches(Object argument) {@SuppressWarnings("unchecked")List<T> list = (List<T>) argument;return list.contains(element);}public static <T> List<T> contains(T element) {return argThat(new ListContainsMatcher<>(element));} }

現在進行一次測試,以展示我們的新Matcher!

@RunWith(MockitoJUnitRunner.class) public class ListContainsMatcherTest {public interface TestClass {public boolean usesStrings(List<String> list);public boolean usesIntegers(List<Integer> list);}private List<String> stringList = Arrays.asList("Hello", "Java", "Code", "Geek");private List<Integer> integerList = Arrays.asList(1, 2, 3, 4, 5);@MockTestClass test;@Testpublic void test() throws Exception {when(test.usesStrings(contains("Java"))).thenReturn(true);when(test.usesIntegers(contains(5))).thenReturn(true);assertTrue(test.usesIntegers(integerList));assertTrue(test.usesStrings(stringList));Mockito.reset(test);when(test.usesStrings(contains("Something Else"))).thenReturn(true);when(test.usesIntegers(contains(42))).thenReturn(true);assertFalse(test.usesStrings(stringList));assertFalse(test.usesIntegers(integerList));Mockito.reset(test);} }

作為練習,嘗試編寫自己的Matcher,如果Map包含特定的鍵/值對,則Matcher將匹配。

6.間諜和部分存根

如前所述,可以使用@Spy批注對類進行部分存根。 部分存根允許我們在測試中使用真實的類,而僅存根與我們有關的特定行為。 Mockito準則告訴我們,在處理遺留代碼時,通常應謹慎偶爾使用間諜。 最佳實踐不是使用Spy部分模擬受測類,而是部分模擬依賴項。 被測類應始終是真實對象。

假設我們正在處理一個在java.awt.BufferedImage上工作的圖像處理類。 此類將BufferedImage放入其構造函數中,并公開一個方法,該方法使用隨機彩色的垂直條紋填充圖像,并根據輸入的縮略圖高度返回圖像的縮略圖。

public class ImageProcessor {private BufferedImage image;public ImageProcessor(BufferedImage image) {this.image = image;}public Image overwriteImageWithStripesAndReturnThumbnail(int thumbHeight) {debugOutputColorSpace();Random random = new Random();Color color = new Color(random.nextInt(255), random.nextInt(255), random.nextInt(255));for (int x = 0; x < image.getWidth(); x++) {if (x % 20 == 0) {color = new Color(random.nextInt(255), random.nextInt(255), random.nextInt(255));for (int y = 0; y < image.getHeight(); y++) {image.setRGB(x, y, color.getRGB());}}}Image thumbnail = image.getScaledInstance(-1, thumbHeight, Image.SCALE_FAST);Image microScale = image.getScaledInstance(-1, 5, Image.SCALE_DEFAULT);debugOutput(microScale);return thumbnail;}private void debugOutput(Image microScale) {System.out.println("Runtime type of microScale Image is " + microScale.getClass());}private void debugOutputColorSpace() {for (int i=0; i< image.getColorModel().getColorSpace().getNumComponents(); i++) {String componentName = image.getColorModel().getColorSpace().getName(i);System.out.println(String.format("Colorspace Component[%d]: %s", i, componentName));}} }

overwriteImageWithStripesAndReturnThumbnail()方法中發生了很多事情。 它要做的第一件事是輸出一些有關圖像顏色空間的調試信息。 然后,它會使用圖像的寬度和高度方法生成一些隨機顏色,并將其繪制為整個圖像中的水平條紋。 然后,它執行縮放操作以返回代表縮略圖的圖像。 然后,它執行第二次縮放操作以生成一個小的診斷微映像,并輸出此微映像的運行時類類型作為調試信息。

我們看到了與BufferedImage的許多交互,其中大多數是完全內部的或隨機的。 最終,當我們要驗證方法的行為時,對我們來說重要的是對getScaledInstance()的首次調用–如果我們方法的返回值是從getScaledInstance()返回的對象,則類可以工作。 這是BufferedImage的行為,它對我們來說很重要。 我們面臨的問題是對BufferedImages方法還有許多其他調用。 從測試的角度來看,我們并不真正在乎這些方法的返回值,但是如果我們不對它們的行為進行編碼,則它們將以某種方式導致NullPointerException并可能導致其他不良行為。

為了解決這個問題,我們將為BufferedImage創建一個Spy,并且僅對我們感興趣的getScaledInstance()方法進行存根處理。

讓我們創建一個空的測試類,其中包含被測類和Spy類,以及一個用于返回縮略圖的Mock。

@RunWith(MockitoJUnitRunner.class) public class ImageProcessorTest {private ImageProcessor processor;@Spyprivate BufferedImage imageSpy = new BufferedImage(800, 600, BufferedImage.TYPE_INT_ARGB);@MockImage mockThumbnail;@Beforepublic void setup() {processor = new ImageProcessor(imageSpy);} }

請注意,BufferedImage沒有默認構造函數,因此我們必須使用它的參數化構造函數自行實例化它,如果它具有默認構造函數,我們可以讓Mockito為我們實例化它。

現在,讓我們首先嘗試暫存我們感興趣的行為。忽略輸入高度,寬度和模式并繼續對所有三個參數使用Argument Matchers是有意義的。 我們最終得到如下內容:

given(imageSpy.getScaledInstance(anyInt(), anyInt(), anyInt())).willReturn(mockThumbnail);

通常,這將是對Spy進行存根的最佳方法,但是,在這種情況下會出現問題– imageSpy是真正的BufferedImage,并且傳遞given() Given given()的存根調用是在存根操作執行時實際執行的真實方法調用由JVM運行。 getScaledInstance要求width和height不為零,因此此調用將導致引發IllegalArgumentException 。

一種可能的解決方案是在存根調用中使用實參

@Testpublic void scale_should_return_internal_image_scaled() throws Exception {// Givengiven(imageSpy.getScaledInstance(-1, 100, Image.SCALE_FAST)).willReturn(mockThumbnail);// WhenImage actualImage = processor.overwriteImageWithStripesAndReturnThumbnail(100);// ThenassertEquals(actualImage, mockThumbnail);}

該測試成功運行,并在控制臺上產生以下輸出

Colorspace Component[0]: Red Colorspace Component[1]: Green Colorspace Component[2]: Blue Runtime type of microScale Image is class sun.awt.image.ToolkitImage

使用實值的副作用是對getScaledInstance()的第二次調用getScaledInstance()用于創建用于調試的微圖像)無法匹配,并且此時執行了BufferedImage中的real方法,而不是我們的存根行為–這就是為什么我們看到真正的輸出的微圖像的運行時類型,而不是Mockito模擬實現,我們將查看是否將嘲笑縮略圖傳遞給了調試輸出方法。

但是,如果我們想繼續使用參數匹配器怎么辦? 可以使用doReturn()方法(如果您記得,通常用于void方法)對getScaledInstance()方法進行存根,而無需在存根時實際調用它。

@Testpublic void scale_should_return_internal_image_scaled_doReturn() throws Exception {// GivendoReturn(mockThumbnail).when(imageSpy).getScaledInstance(anyInt(), anyInt(), anyInt());// WhenImage actualImage = processor.overwriteImageWithStripesAndReturnThumbnail(100);// ThenassertEquals(actualImage, mockThumbnail);}

這給出以下輸出:

Colorspace Component[0]: Red Colorspace Component[1]: Green Colorspace Component[2]: Blue Runtime type of microScale Image is class $java.awt.Image$$EnhancerByMockitoWithCGLIB$$72355119

您可以看到微映像的運行時類型現在是Mockito創建的Mock實現。 之所以如此,是因為兩個對getScaledInstance調用getScaledInstance與存根參數匹配,因此兩個調用都返回了Mock縮略圖。

有一種方法可以確保在第二個實例中調用Spy的真實方法,方法是使用doCallRealMethod()方法。 像往常一樣,Mockito讓您將存根方法鏈接在一起,以便為與存根參數匹配的存根方法的連續調用編寫不同的行為。

@Testpublic void scale_should_return_internal_image_scaled_doReturn_doCallRealMethod() throws Exception {// GivendoReturn(mockThumbnail).doCallRealMethod().when(imageSpy).getScaledInstance(anyInt(), anyInt(), anyInt());// WhenImage actualImage = processor.overwriteImageWithStripesAndReturnThumbnail(100);// ThenassertEquals(actualImage, mockThumbnail);}

給出以下輸出

Colorspace Component[0]: Red Colorspace Component[1]: Green Colorspace Component[2]: Blue Runtime type of microScale Image is class sun.awt.image.ToolkitImage

7.結論

我們已經研究了許多針對嘲笑和間諜的舉止行為的方式,并且暗示了人們可以舉止行為的方式幾乎無限。

Mockito的javadoc是有關Stubbing方法(尤其是Mockito開箱即用提供的ArgumentMatchers)的良好信息來源。

我們已經詳細介紹了存根行為,在下一個教程中,我們將研究使用Mockito驗證框架來驗證Mocks的行為。

8.下載源代碼

這是關于Mockito Stubbing的課程。 您可以在此處下載源代碼: mockito2-stubbing

翻譯自: https://www.javacodegeeks.com/2015/11/mocks-spies-partial-mocks-and-stubbing.html

cks32和stm32

創作挑戰賽新人創作獎勵來咯,堅持創作打卡瓜分現金大獎

總結

以上是生活随笔為你收集整理的cks32和stm32_cks子,间谍,局部Mo子和短管的全部內容,希望文章能夠幫你解決所遇到的問題。

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

国产精品久免费的黄网站 | 成人av免费在线观看 | 午夜精品久久久久久 | 中文字幕有码在线 | www.夜夜爱 | 天天色影院 | 狠狠干我 | 亚洲精品麻豆视频 | 日一日干一干 | 麻豆国产网站 | 欧美五月婷婷 | 精品国产乱码久久久久久浪潮 | 亚洲精品国产第一综合99久久 | 欧美视频网址 | 91av蜜桃| 免费黄色av | 久久久久99精品国产片 | 在线看片中文字幕 | 日韩av午夜在线观看 | 精品二区久久 | 五月婷婷丁香在线观看 | 国产伦理久久 | 久在线观看视频 | 成片免费观看视频大全 | 国产手机视频 | 亚洲精品福利在线观看 | 亚洲精品黄色 | 国产成在线观看免费视频 | 婷婷综合亚洲 | 狠狠色狠狠色合久久伊人 | 亚洲午夜av电影 | 91成人在线网站 | 国产午夜精品免费一区二区三区视频 | 国产成人一区三区 | 99热最新精品| 亚洲影院国产 | 九九九热精品免费视频观看 | 在线观看完整版免费 | 最新av网址在线观看 | 亚洲国产三级在线观看 | 又黄又爽又湿又无遮挡的在线视频 | 美女国内精品自产拍在线播放 | 久久96国产精品久久99漫画 | 国产分类视频 | 国产成人精品福利 | 一区中文字幕在线观看 | 精品国产一区二区三区久久久蜜臀 | 中文字幕日本在线观看 | 国产永久免费高清在线观看视频 | 中文字幕中文字幕在线一区 | 日韩最新理论电影 | 蜜桃av综合网 | 国产精品毛片久久久久久久久久99999999 | 欧美日韩在线观看一区 | 97超碰成人 | 麻豆免费视频 | 91在线播放视频 | 一区二区视频播放 | 欧美日韩国产伦理 | 国产特级毛片aaaaaa毛片 | 亚洲国产精品第一区二区 | 欧美小视频在线观看 | 97超碰中文 | 91黄在线看| 韩国av免费在线观看 | 精品一区二区在线观看 | 国产福利一区二区三区视频 | 91天天操 | 国产亚洲精品中文字幕 | 美女精品网站 | 亚洲日本一区二区在线 | 亚洲aⅴ久久精品 | 久久免费国产精品 | 久草在线99 | 久久久免费av | 中文在线亚洲 | 日韩91在线 | 日韩精品久久久久久 | 91在线免费视频 | 日韩va欧美va亚洲va久久 | 久久激情视频 久久 | 亚洲视频免费在线观看 | 欧美一级在线观看视频 | 亚洲成a人片77777潘金莲 | 午夜在线观看一区 | 日韩免费在线视频观看 | 久久综合成人网 | 久久视频热 | 三三级黄色片之日韩 | 九九九九精品九九九九 | 69精品在线观看 | 免费观看性生活大片 | 一区在线免费观看 | 香蕉视频网站在线观看 | 国产一级片观看 | 精品在线观看国产 | 色a资源在线 | 欧美在线观看禁18 | 国产精品久久久久久久免费大片 | 99免费在线播放99久久免费 | 亚洲国产天堂av | 亚洲综合网站在线观看 | 日韩欧美专区 | 国产黄免费看 | 国产不卡视频在线播放 | 国产在线观看地址 | 成 人 黄 色 视频 免费观看 | 国产成人三级一区二区在线观看一 | 天天色综合天天 | 精品免费在线视频 | 欧美激情精品久久久久久变态 | 国产69久久久 | 国产精品免费久久久久影院仙踪林 | 国色天香在线观看 | 一色av | 欧美在线观看禁18 | 国产精品久久久久久久久久不蜜月 | 操高跟美女 | 国产手机视频在线观看 | 色综合久久五月天 | 99999精品| 最新色站 | 亚洲国产精品久久久 | 国产手机精品视频 | 精品国产一区二区三区在线 | 免费在线国产黄色 | 日韩av综合网站 | 欧亚日韩精品一区二区在线 | 欧洲色综合 | 亚洲三区在线 | 色丁香婷婷 | 黄色毛片一级 | 精品视频免费在线 | 韩国一区视频 | 九九免费观看全部免费视频 | 五月情婷婷 | 91福利试看 | 国产小视频在线观看 | 欧美另类v | 免费欧美高清视频 | 午夜精品一区二区三区在线观看 | 国产视频黄 | 五月婷婷丁香 | 亚洲成a人片77777kkkk1在线观看 | 天天干天天碰 | 久久伦理视频 | 国产成人精品在线观看 | 国产日韩中文字幕 | 伊人五月婷 | 日韩免费在线观看视频 | 欧美日韩视频 | 国产免费一区二区三区最新6 | 国产成人一区三区 | 日韩一级黄色片 | 国产区免费在线 | 国产乱码精品一区二区蜜臀 | 国产黄在线播放 | 韩国精品福利一区二区三区 | 免费看片网址 | 成人av在线影院 | a级免费观看 | 国产打女人屁股调教97 | 波多野结依在线观看 | 国产一区在线免费观看视频 | 久久一区二区三区国产精品 | 午夜精选视频 | 精品国产1区二区 | 成人欧美日韩国产 | 欧美日韩免费看 | 日韩三级久久 | 久久专区| 在线免费观看黄色小说 | 亚洲乱码在线 | 久久久精品久久日韩一区综合 | 成年一级片 | 亚洲一区二区三区毛片 | 免费视频二区 | 国产精品美女免费视频 | 国产尤物在线观看 | 欧美色图亚洲图片 | 精品久久福利 | 免费在线观看不卡av | 97天天干 | 黄色大全免费网站 | 国产亚洲成人网 | 午夜精品久久久久久99热明星 | 在线免费观看羞羞视频 | 黄色av电影 | 在线日韩精品视频 | 日日夜色 | 日本不卡123区 | 国产精品6| 日韩在线视频免费播放 | 激情av网址 | 精品视频 | 人人爽人人爽人人 | 欧美污在线观看 | 久久国产乱| 国产资源免费 | 欧美精品色 | 日韩r级在线 | 欧美在线视频一区二区三区 | 国产精品一区免费看8c0m | 午夜精品999| 国产精品18久久久久久vr | 久久久久久久久久久福利 | 国内外激情视频 | 最近中文字幕国语免费av | 免费成视频| 亚洲精品福利在线 | 99国产精品视频免费观看一公开 | 久久91网| 免费人成在线观看 | 在线看一区| 一区在线电影 | 欧美精品久久天天躁 | 欧美污污网站 | 中文字幕文字幕一区二区 | 91视频免费网站 | 亚洲一区二区三区在线看 | 97超碰在线久草超碰在线观看 | 六月丁香激情综合 | 天天色.com | 久久综合综合久久综合 | 亚洲精品动漫成人3d无尽在线 | 天天天综合 | 国产97视频在线 | .国产精品成人自产拍在线观看6 | 国产一区二区不卡在线 | 啪啪免费视频网站 | 中文有码在线 | 91在线中字 | 精品中文字幕在线 | 超碰在线97免费 | 日韩视频免费观看高清完整版在线 | 18国产精品白浆在线观看免费 | 黄色小说网站在线 | 色97在线| 中文字幕在线观看视频一区 | 国产品久精国精产拍 | 亚洲黄色激情小说 | 97日日碰人人模人人澡分享吧 | 午夜少妇 | av网站在线免费观看 | 中文字幕高清在线 | 久久色视频 | 久久精品一区二区三区视频 | 成人一区电影 | 亚洲欧美日韩国产一区二区 | 91精品亚洲影视在线观看 | 少妇bbbb搡bbbb桶| 天天操天天色天天 | 日韩最新在线视频 | 久久综合狠狠 | 免费国产亚洲视频 | 成年人免费在线看 | www.五月天 | 91精品视频在线免费观看 | 亚洲欧美一区二区三区孕妇写真 | 伊人在线视频 | 色婷婷 亚洲 | 婷婷激情5月天 | 成人小视频在线观看免费 | 在线观看v片 | 国产精品男女视频 | 一区二区三区观看 | 亚洲高清视频在线观看 | 久久精品国产免费看久久精品 | 在线免费高清一区二区三区 | 2017狠狠干 | 91在线麻豆 | 欧美aa一级 | av中文字幕免费在线观看 | 亚洲黄色免费电影 | 国产福利一区二区在线 | 中文字幕在线免费看线人 | 午夜精品一区二区三区免费 | 91欧美精品 | 成人黄色在线 | 久草av在线播放 | 国产中文字幕一区二区三区 | 91黄视频在线 | 欧美aaa级片 | 精品久久1 | 久久99热久久99精品 | 色网站在线免费 | 欧美在线不卡一区 | 欧美成人精品在线 | 日日干av | 色国产精品一区在线观看 | 99久久婷婷国产一区二区三区 | 性色va | 久久国产二区 | 国产精品亚洲人在线观看 | 国产精品2019 | 免费在线观看亚洲视频 | 在线看免费 | 在线观看国产一区二区 | 在线播放91 | 免费看一级片 | 欧美日韩在线播放 | 国产精品爽爽久久久久久蜜臀 | 国产精品九九九九九 | 99热精品免费观看 | 天天爽夜夜操 | 免费观看丰满少妇做爰 | 日韩av高清 | 国产精品av在线 | 午夜色性片| 国产成人一区在线 | 黄污视频大全 | 久久黄色片子 | 久久久久免费精品国产 | 狠狠干天天色 | 日韩三级不卡 | 亚洲视频axxx | 国产成人综合在线观看 | 国产精品免费看久久久8精臀av | 欧美日韩精品在线 | 婷婷五月色综合 | 国产在线色视频 | 天天做日日爱夜夜爽 | 免费黄色在线播放 | 黄色一区二区在线观看 | 国产群p视频 | 亚洲国产成人久久 | 色婷婷国产精品 | 久久久久久久久毛片精品 | 青草视频网| www成人av| 国产一级免费在线 | 国产精品久久久久久一二三四五 | 色婷婷激婷婷情综天天 | 日韩av中文字幕在线免费观看 | 国产成人精品999 | av高清一区二区三区 | 日韩高清在线不卡 | 在线v片免费观看视频 | 国产小视频你懂的在线 | 久久久久国产视频 | 欧美在线资源 | 国产.精品.日韩.另类.中文.在线.播放 | 亚洲伊人网在线观看 | 国产一区在线观看免费 | 不卡视频国产 | 免费av在线播放 | 在线91精品 | 国产色视频网站 | 久草精品视频在线看网站免费 | 三级黄色大片在线观看 | 四虎天堂 | 免费av在线播放 | 丁香久久婷婷 | 97av视频在线观看 | av电影在线免费观看 | 久久99久久99精品免观看软件 | 成人免费在线观看入口 | 毛片永久新网址首页 | 国产护士在线 | 麻豆网站免费观看 | 一区二区中文字幕在线 | 91九色蝌蚪国产 | 欧美日韩在线播放 | 久久婷婷一区 | 久久只精品99品免费久23小说 | 国产精品观看在线亚洲人成网 | www.福利视频| 国产成人亚洲精品自产在线 | 国产在线高清 | 精品亚洲免费 | 国产一级一片免费播放放 | 国产一区播放 | 亚洲japanese制服美女 | 久久影院中文字幕 | 波多野结衣在线中文字幕 | 91看片淫黄大片91 | av大全在线看 | 婷婷久久五月 | 久一久久 | 99精品一级欧美片免费播放 | 日韩欧美视频在线观看免费 | 久久99精品视频 | 亚洲更新最快 | www.夜夜骑.com | 天天久久夜夜 | 亚洲国产99 | 天天操天天操天天爽 | 国产精品免费视频网站 | 亚洲在线看 | 在线观看视频福利 | 国产精品理论片在线播放 | 欧美乱大交 | 激情综合网色播五月 | 婷婷亚洲综合 | 精品国产免费看 | 天天爽天天摸 | 91在线精品视频 | 麻豆精品在线视频 | 91在线超碰| 国产精品久久久久四虎 | 免费在线观看av网址 | 日韩一二区在线观看 | 久久综合综合久久综合 | 欧美激情视频一二区 | 亚洲精品午夜久久久久久久久久久 | 91黄在线看| 国产高清免费在线观看 | 美女网站色免费 | 成人中心免费视频 | 国产亚洲精品久久久久久电影 | 九九视频一区 | 亚洲日本va午夜在线影院 | 亚洲天堂网站视频 | 中文字幕a∨在线乱码免费看 | 精品国产一区二区三区久久影院 | 日韩欧美亚州 | www.天天草| 日日夜夜精品免费视频 | 久久999精品 | 一级黄色a视频 | 国产亚洲在线视频 | 亚洲成年人免费网站 | 日韩在线不卡av | 国产精品久久三 | 免费在线观看av不卡 | 最近最新中文字幕 | 美女免费视频一区 | 91成人免费观看视频 | 久精品视频在线观看 | 亚州精品视频 | 日韩欧美网址 | 国产成人精品女人久久久 | 亚洲精品国产自产拍在线观看 | 久久综合偷偷噜噜噜色 | 精品久久精品久久 | 中文字幕日韩伦理 | 精品国产乱码久久久久久天美 | 免费在线播放视频 | 国产人成看黄久久久久久久久 | 久久国产一区 | 蜜桃麻豆www久久囤产精品 | 婷婷综合久久 | 黄色网在线播放 | 日韩成人看片 | 91在线免费公开视频 | 久久av免费 | 国产视频精品在线 | 手机版av在线 | 91在线资源 | 欧美日韩视频在线观看一区二区 | 人人澡超碰碰 | 国产四虎影院 | 中文字幕美女免费在线 | 中日韩在线| 国产精品乱码久久 | 日本丶国产丶欧美色综合 | 黄色av成人在线观看 | 综合久久久久久久久 | 麻豆影视在线播放 | www.狠狠操.com | 国产麻豆果冻传媒在线观看 | 精品视频123区在线观看 | 在线免费观看国产精品 | 天天操天天操天天操天天操天天操天天操 | 91免费观看网站 | 亚洲欧洲日韩 | 亚洲欧洲视频 | 天天操天天是 | 波多野结衣在线播放视频 | 九九国产精品视频 | 免费福利片2019潦草影视午夜 | 青青河边草观看完整版高清 | 亚洲黄色在线 | 九九九电影免费看 | 99视频在线免费观看 | 婷婷成人亚洲综合国产xv88 | 国产一区二区三区免费视频 | 欧美日韩在线精品一区二区 | 黄色毛片视频免费 | 国产精品久久嫩一区二区免费 | 在线观看国产www | 久草免费在线 | 一本色道久久综合亚洲二区三区 | 亚洲毛片在线观看. | 久久社区视频 | 一区二区三区精品在线视频 | 色婷婷久久一区二区 | 伊人久久av | 日韩欧美视频免费看 | 国产夫妻自拍av | 色五月情| 天天干天天上 | 久久精品一区二区三区中文字幕 | 中文字幕久久精品亚洲乱码 | 精品夜夜嗨av一区二区三区 | 日本九九视频 | 精品亚洲欧美无人区乱码 | 国产精品原创 | 国产精品久久久久久久久久尿 | 91九色最新地址 | 色综合天天综合 | 99免费视频| 欧美国产91 | 91九色视频在线 | 日本三级大片 | 中文字幕av影院 | 一区二区视频在线观看免费 | 婷婷亚洲五月色综合 | 99视频在线精品国自产拍免费观看 | 亚洲欧美在线观看视频 | 正在播放 国产精品 | 五月色综合| 久久久久久久久久久免费av | 丁香在线视频 | 亚洲国产日本 | 九九视频在线 | 免费在线黄色av | 久久精品国产免费看久久精品 | 97成人精品区在线播放 | 在线免费观看国产 | 麻豆国产网站入口 | www.超碰| 色国产精品一区在线观看 | av成人免费网站 | 久久精品人人做人人综合老师 | 国产精品毛片一区视频 | 蜜臀精品久久久久久蜜臀 | 国内精品久久久久久 | 九九免费观看全部免费视频 | 欧美黑人性猛交 | 欧美色噜噜 | 日韩a欧美 | 亚洲成年人av | 少妇bbw撒尿 | 在线观看国产区 | 色综合久久久久久中文网 | 岛国大片免费视频 | av三级在线免费观看 | 欧美日韩精品在线 | 夜夜骑日日操 | 欧美一级乱黄 | 国产又黄又爽又猛视频日本 | 成人高清av在线 | 在线 视频 一区二区 | 久久久国产精华液 | 日韩欧美精品在线 | 中文字幕一区二 | 国产精品一区在线播放 | 蜜桃麻豆www久久囤产精品 | 亚洲 欧美 综合 在线 精品 | 天天插天天爽 | 在线观看免费 | 国产亚洲在线视频 | 国产精品久久久久久五月尺 | 日日干夜夜干 | 黄色av大片| 国产一区二区在线播放视频 | 久久亚洲国产精品 | 国产精品99爱| 日韩三级视频 | 欧美精品在线观看免费 | 久久综合免费视频影院 | 亚洲作爱视频 | 日韩av不卡在线 | 黄色成人91| 中文字幕一区二区三区久久蜜桃 | 国产精品久久久久久久久久免费 | 国语精品免费视频 | 麻豆视频一区 | 黄色日视频| 欧美日韩三级在线观看 | 欧美少妇影院 | 国产高清视频免费观看 | 精品视频网站 | 久射网| 欧美成年黄网站色视频 | 四虎成人网 | 国产精品一区二区av影院萌芽 | 成人资源站| 亚洲欧美日韩精品一区二区 | 国产999精品久久久久久绿帽 | 一级黄色在线免费观看 | 香蕉视频国产在线 | 伊人色综合久久天天网 | 91porny九色在线播放 | 国产精品久久99精品毛片三a | 成全在线视频免费观看 | 免费网站观看www在线观看 | 五月婷婷丁香综合 | www.日本色 | 最新国产中文字幕 | 日韩av一区二区在线播放 | 三级av免费看 | 曰本三级在线 | 丁香婷婷成人 | 国产高清精 | 成人免费看视频 | 久久一区精品 | 99免费在线视频 | 免费一区在线 | 96精品高清视频在线观看软件特色 | 一本一本久久aa综合精品 | 欧美一区三区四区 | 成人黄色大片 | 国产区精品在线观看 | 香蕉视频一级 | 久久婷婷久久 | 亚洲视频1区2区 | 亚洲日日日 | 亚洲高清在线观看视频 | 亚洲狠狠 | 欧美一区二区三区在线视频观看 | 婷婷色中文 | 波多野结衣日韩 | www.狠狠操 | 在线黄色免费av | 日韩电影在线视频 | 精品国产一区二区三区在线 | 日本韩国欧美在线观看 | 久久首页 | 波多野结衣小视频 | 久久夜色精品亚洲噜噜国4 午夜视频在线观看欧美 | 国产无遮挡猛进猛出免费软件 | 中文字幕在线观看完整版电影 | 婷婷色综合网 | 人人干干人人 | 欧美日韩高清一区二区 | 亚洲激情视频在线观看 | 99精品视频一区二区 | 天天操天天添 | 18久久久 | 国产精品1区2区3区 久久免费视频7 | 国产一区二区三区免费观看视频 | 国产破处精品 | 黄色大片入口 | 91成人在线免费观看 | 最新av免费在线 | 免费看国产黄色 | 亚洲伊人婷婷 | 久久综合欧美 | 91在线视频网址 | 国产你懂的在线 | 亚洲三级黄 | 欧美日韩国产三级 | 国产自在线| 亚洲激情影院 | 国产一区二区影院 | 免费看久久久 | 国产精品久久电影观看 | 欧美成年黄网站色视频 | 在线视频观看亚洲 | 国产中文字幕在线播放 | 国产精品18久久久久vr手机版特色 | 99久久er热在这里只有精品15 | 国产a国产 | 中文字幕一区二区三区四区视频 | 在线视频欧美日韩 | 国产拍在线 | 国际精品网| 久久免费99精品久久久久久 | www.婷婷色| 午夜精品视频免费在线观看 | 看全黄大色黄大片 | 天天看天天操 | 日韩av成人在线观看 | 在线观看你懂的网站 | 久久久久激情 | 丁香av在线 | 日本精品久久久久久 | 成人免费视频播放 | 九九在线精品视频 | 亚洲国产成人精品电影在线观看 | 免费精品在线 | 91精品国产92久久久久 | 色视频网站在线 | 九九九九热精品免费视频点播观看 | 正在播放一区 | 欧美日韩国内在线 | 国内精品久久久久影院优 | 婷婷伊人五月天 | 亚洲国产中文在线 | 最近日本中文字幕a | 99视频在线免费 | 欧美美女视频在线观看 | 国产精品网红福利 | 伊人亚洲综合网 | 日韩欧美黄色网址 | 人人插人人爱 | 国产二区av| 色网站黄 | 超碰免费在线公开 | 国产高清av| 亚洲午夜久久久久久久久久久 | 成人av.com | 91在线小视频 | 欧美做受69 | 天天射天天爱天天干 | 999电影免费在线观看2020 | 欧美一级性 | 99视频精品在线 | 日韩在线观看电影 | 精品91久久久久 | 一级片黄色片网站 | 亚洲视频网站在线观看 | 亚洲最新av在线 | 色久综合 | 国产精品久久久久aaaa九色 | 国产午夜在线观看视频 | 精品中文字幕视频 | 欧美大片www | 最新精品视频在线 | 国产黄色大片 | 亚洲女人天堂成人av在线 | 91丨九色丨高潮丰满 | 少妇性aaaaaaaaa视频 | 娇妻呻吟一区二区三区 | av一级在线观看 | 六月天色婷婷 | 69av在线播放 | 中文字幕在线观看免费高清电影 | 日本黄色片一区二区 | 欧美一级电影免费观看 | 久久色网站 | 免费在线国产精品 | 久久免费视频在线 | 黄色精品久久 | 国产亚洲精品久久久久久网站 | 精品一区二区三区在线播放 | 视色网站 | 日日操天天操狠狠操 | 福利视频导航网址 | 天天天天天天天天操 | 成人av高清在线 | www.亚洲| 国产成人综合图片 | 国产精品久久99综合免费观看尤物 | 久久精品999| 国产精品99免费看 | 亚洲精品视频二区 | 久精品视频在线 | 少妇自拍av| 国产一区二区三区午夜 | 国产黄在线免费观看 | 性色av免费在线观看 | 91免费高清视频 | 91人人揉日日捏人人看 | 人人澡超碰碰97碰碰碰软件 | 国产在线高清精品 | 91视频免费看 | 成人a大片 | 国产精品1区2区 | 五月天久久久久 | 一区电影| 麻花豆传媒mv在线观看 | 色中文字幕在线观看 | 色婷婷免费视频 | 日本特黄一级 | 免费看黄在线 | 国产精品九九九 | 欧美日韩在线观看不卡 | 成人免费视频网 | 一区二区视频网站 | 婷婷色五 | 欧美一二三视频 | 日韩免费视频播放 | 国产色在线 | 狠狠狠狠狠狠狠 | 日韩精品欧美精品 | 婷婷亚洲五月 | 国产 字幕 制服 中文 在线 | 亚洲在线综合 | 深爱激情站 | 久久精品视频在线观看 | 色黄www小说| 久久国产网 | 免费日韩电影 | 91在线免费播放视频 | 国产精品久久久久婷婷二区次 | 国产黄色大全 | 一区二区三区av在线 | 久久99国产精品久久 | 亚洲v欧美v国产v在线观看 | 国产精品青草综合久久久久99 | 久久综合九色综合网站 | 黄色成人影视 | 久久免费视频在线观看6 | 插婷婷 | 免费日韩 精品中文字幕视频在线 | 97在线观看免费观看高清 | 丁香综合网 | 91在线视频观看 | 欧美 日韩 国产 中文字幕 | 精品国产一区二区三区久久久蜜臀 | 亚洲一区二区黄色 | 国产精品不卡一区 | 欧美整片sss | 区一区二区三在线观看 | 亚洲a资源| 久久调教视频 | 国产成人精品一区二区三区网站观看 | 婷婷色狠狠 | 日韩高清无线码2023 | 国产字幕在线观看 | 国产又粗又硬又长又爽的视频 | 天天射狠狠干 | 麻豆传媒视频在线免费观看 | 国产精品久久精品 | 日产乱码一二三区别在线 | 成年人在线免费看视频 | 一区二区 不卡 | 黄色片免费在线 | 欧美日韩三级在线观看 | 国产精品夜夜夜一区二区三区尤 | 美女av在线免费 | av播放在线 | 日韩在线资源 | 成人久久免费 | 天堂中文在线播放 | 免费黄a大片 | 国产精品av久久久久久无 | 国产一区二区三区高清播放 | 99国内精品 | 日韩av一区在线观看 | 久久久男人的天堂 | 久久午夜精品视频 | 成年人在线免费视频观看 | 99视频在线观看一区三区 | 欧美俄罗斯性视频 | 日日夜夜精品 | 天堂在线视频中文网 | 日韩在线中文字幕视频 | 久久伦理影院 | 欧美一级免费片 | 国产青青青 | 免费在线观看亚洲视频 | 99久久久久国产精品免费 | 中文字幕在线观看网站 | 日韩欧美中文 | 一区二区三区在线免费播放 | 免费观看一级一片 | 国产精品一区欧美 | 欧美一级大片在线观看 | 最新日韩中文字幕 | 婷婷色中文网 | 日韩高清免费在线观看 | 国产亚洲欧美精品久久久久久 | www.xxxx欧美| 国产系列在线观看 | 国模视频一区二区三区 | 欧美在线一二区 | 亚洲精品国偷拍自产在线观看蜜桃 | 99久精品视频 | 91亚洲精品久久久久图片蜜桃 | 夜夜操天天摸 | 国产免费一区二区三区网站免费 | 天天摸天天舔 | 日日麻批40分钟视频免费观看 | 免费看色的网站 | 丝袜美腿亚洲 | 黄色日批网站 | 欧美激情综合五月色丁香 | 中文字幕免费高清 | 免费高清在线观看成人 | 国产一区二区精品91 | 五月婷在线观看 | 国产成人av片 | 国产只有精品 | 亚洲精品人人 | 又色又爽又激情的59视频 | 午夜日b视频| 西西44人体做爰大胆视频 | 中文字幕国产精品一区二区 | 国产欧美精品一区二区三区四区 | 久久深爱网| 激情网站网址 | av官网在线| 在线91av| 中文字幕黄色av | 亚洲国产中文字幕在线视频综合 | 亚洲欧美综合精品久久成人 | av在线免费播放网站 | 久久久视频在线 | 国产第一页福利影院 | 久久久久久久久艹 | 91av蜜桃| 国产一区福利在线 | 超碰在线人 | 久久久精品欧美 | www.日本色 | a级成人毛片 | 国内久久精品 | 久久99久久99精品免视看婷婷 | 美女网站免费福利视频 | 中文字幕精品三区 | 欧美日韩91 | 91精品国自产在线观看欧美 | 日日天天av| 天天色宗合 | 亚洲国产精品推荐 | 亚洲va欧美va人人爽春色影视 | 一区二区欧美在线观看 | 日韩一区二区三区在线看 | 色综合久久久 | 成人av资源| 久久国产精品免费一区二区三区 | 99精品视频在线观看免费 | 欧美日本在线视频 | 国产一区欧美二区 | 福利视频网站 | 成年人免费在线观看网站 | 久久久精品网站 | 亚洲一区二区视频在线播放 | 涩涩网站在线观看 | 国产午夜精品福利视频 | 黄色在线观看免费网站 | 四虎永久免费在线观看 | 国产精品综合av一区二区国产馆 | 国产一区二区精品久久91 | 天天射综合网站 | 国产在线精品一区二区 | 91色吧 | 国产精品久久久久久久久久妇女 | 国产精品3| 久久久久久福利 | 日日夜夜精品视频天天综合网 | 日韩精品一区二区三区免费观看视频 | 成人av中文字幕在线观看 | 亚洲一级黄色片 | 色吧av色av | 日韩有码在线播放 | 国产精品久久伊人 | 亚洲一区av | 国产精品色婷婷视频 | 国产在线视频一区二区 | 夜夜天天干 | 免费毛片一区二区三区久久久 | 免费男女羞羞的视频网站中文字幕 | 黄色的视频 | 人人爽久久久噜噜噜电影 | 久久久激情视频 | 九九热只有这里有精品 | 97免费 | 亚洲黄色免费网站 | 欧洲精品一区二区 | 亚洲aⅴ在线 | 成年人在线观看视频免费 | 国产精品99久久久精品免费观看 | 国产色网站 | 射久久| 成人理论电影 | 96亚洲精品久久 | 国产裸体永久免费视频网站 | 精品国产乱码久久久久久天美 | 99视频精品全部免费 在线 | 国产乱老熟视频网88av | 最近免费中文字幕 | 精品在线观看一区二区 | 98久久| 久久精品站| 久久精品国产精品亚洲精品 | 国产一区二区精品久久91 | 中文字幕一区二区三区乱码不卡 | 日批视频| 国产成人精品女人久久久 | 人人舔人人舔 | 免费精品久久久 | 亚洲第一中文网 | 伊人久久精品久久亚洲一区 | av一级片网站 | 亚洲一区二区黄色 | 欧美一级黄色网 | 波多野结衣视频一区二区三区 | 久久久久国产精品视频 | 免费观看视频黄 | 男女激情免费网站 | 色综合天天狠天天透天天伊人 | 亚洲涩涩色 | 欧美做受高潮电影o | 色噜噜日韩精品一区二区三区视频 | 91看片网址 | 午夜在线观看影院 | 色视频国产直接看 | 一区二区欧美激情 | 91久久偷偷做嫩草影院 | 欧美激情精品久久久久久免费印度 | 四虎最新域名 | 免费a视频 | 在线日韩中文字幕 | 99视频国产精品 | 国产成人在线网站 | 日韩视频在线播放 | av一二三区| 一区二区伦理 | 日本中文字幕免费观看 | 国产色婷婷 | 亚洲一区二区三区精品在线观看 | 国产成人亚洲在线观看 | 久久精品在线 |