ทำให้วิธีการเยาะเย้ยกลับอาร์กิวเมนต์ที่ผ่านไป


674

พิจารณาลายเซ็นวิธีการเช่น:

public String myFunction(String abc);

Mockito ช่วยส่งคืนสตริงเดียวกันกับที่ได้รับหรือไม่?


ตกลงแล้วกรอบการเยาะเย้ยของจาวาทั่วไปจะเป็นไปได้อย่างไรกับเฟรมเวิร์กอื่นหรือฉันควรสร้าง stub ที่โง่เพื่อเลียนแบบพฤติกรรมที่ฉันต้องการ
Abhijeet Kashnia

คำตอบ:


1001

คุณสามารถสร้างคำตอบใน Mockito สมมติว่าเรามีอินเทอร์เฟซชื่อแอปพลิเคชันด้วยวิธีการของฟังก์ชัน

public interface Application {
  public String myFunction(String abc);
}

นี่คือวิธีทดสอบพร้อมคำตอบ Mockito:

public void testMyFunction() throws Exception {
  Application mock = mock(Application.class);
  when(mock.myFunction(anyString())).thenAnswer(new Answer<String>() {
    @Override
    public String answer(InvocationOnMock invocation) throws Throwable {
      Object[] args = invocation.getArguments();
      return (String) args[0];
    }
  });

  assertEquals("someString",mock.myFunction("someString"));
  assertEquals("anotherString",mock.myFunction("anotherString"));
}

ตั้งแต่ Mockito 1.9.5 และ Java 8 คุณยังสามารถใช้ lambda expression:

when(myMock.myFunction(anyString())).thenAnswer(i -> i.getArguments()[0]);

1
นี่คือสิ่งที่ฉันกำลังมองหาเช่นกัน ขอบคุณ! แต่ปัญหาของฉันก็แตกต่างกัน ฉันต้องการที่จะเยาะเย้ยบริการการคงอยู่ (EJB) ที่เก็บวัตถุและส่งกลับพวกเขาตามชื่อ
migu

7
ฉันสร้างคลาสพิเศษที่ล้อมรอบการสร้างคำตอบ ดังนั้นรหัสอ่านเช่นwhen(...).then(Return.firstParameter())
SpaceTrucker

69
ด้วย Java 8 lambdas when(foo(any()).then(i -> i.getArgumentAt(0, Bar.class))มันเป็นอาหารมื้อเย็นง่ายที่จะกลับอาร์กิวเมนต์แรกแม้สำหรับเรียนเฉพาะคือ และคุณสามารถใช้วิธีการอ้างอิงและเรียกวิธีการจริงได้เช่นกัน
Paweł Dyda

วิธีนี้จะช่วยแก้ปัญหาของฉันด้วยวิธีการส่งคืนIterator<? extends ClassName>ซึ่งทำให้เกิดปัญหาการโยนทุกชนิดในthenReturn()คำสั่ง
Michael Shopsin

16
ด้วย Java 8 และ Mockito <1.9.5 จากนั้นคำตอบของPawełจะกลายเป็นwhen(foo(any()).thenAnswer(i -> i.getArguments()[0])
Graeme Moss

567

หากคุณมี Mockito 1.9.5 หรือสูงกว่ามีวิธีคงที่ใหม่ที่สามารถสร้างAnswerวัตถุให้คุณได้ คุณต้องเขียนอะไรซักอย่าง

import static org.mockito.Mockito.when;
import static org.mockito.AdditionalAnswers.returnsFirstArg;

when(myMock.myFunction(anyString())).then(returnsFirstArg());

หรืออีกวิธีหนึ่ง

doAnswer(returnsFirstArg()).when(myMock).myFunction(anyString());

โปรดทราบว่าreturnsFirstArg()วิธีการนี้เป็นแบบสแตติกในAdditionalAnswersคลาสซึ่งเป็นสิ่งใหม่สำหรับ Mockito 1.9.5 ดังนั้นคุณจะต้องนำเข้าคงที่ที่เหมาะสม


17
หมายเหตุ: มันwhen(...).then(returnsFirstArg())ฉันผิดwhen(...).thenReturn(returnsFirstArg())ที่ให้java.lang.ClassCastException: org.mockito.internal.stubbing.answers.ReturnsArgumentAt cannot be cast to
เบเนดิกคโคพเพล

1
หมายเหตุ: returnFirstArg () ส่งคืนคำตอบ <> แทนค่าของอาร์กิวเมนต์ มี 'Foo (java.lang.String) ไม่สามารถนำไปใช้กับ' (org.mockito.stubbing.Answer <java.lang.Object>) 'ในขณะที่พยายามเรียก. thRRurn (ใหม่ Foo (returnFirstArg ())
Lu55

ฉันต้องการ google คำตอบนี้ซ้ำแล้วซ้ำเล่าในช่วงหลายปีที่ผ่านมาเพราะฉันจำไม่ได้ว่า "ผู้ตอบคำถามเพิ่มเติม" และฉันต้องการมันน้อยมาก จากนั้นฉันก็สงสัยว่า heck ที่ฉันสามารถสร้างสถานการณ์นั้นได้อย่างไรในเมื่อฉันไม่พบสิ่งที่ต้องพึ่งพา ไม่สามารถเพิ่มสิ่งนี้ลงใน mockito ได้โดยตรงหรือ : /
BAERUS

2
คำตอบของสตีฟนั้นธรรมดากว่ามาก อันนี้ให้คุณกลับอาร์กิวเมนต์ดิบเท่านั้น หากคุณต้องการประมวลผลอาร์กิวเมนต์นั้นและส่งคืนผลลัพธ์ดังนั้นกฎการตอบของ Steve ฉันโหวตทั้งคู่เพราะทั้งคู่มีประโยชน์
akostadinov

FYI static org.mockito.AdditionalAnswers.returnsFirstArgเราจะต้องนำเข้า สิ่งนี้เพื่อใช้ returnFirstArg นอกจากนี้ฉันสามารถทำได้when(myMock.myFunction(any())).then(returnsFirstArg())ใน Mockito 2.20 *
gtiwari333

78

ด้วย Java 8 เป็นไปได้ที่จะสร้างคำตอบแบบบรรทัดเดียวแม้กับ Mockito เวอร์ชันเก่า:

when(myMock.myFunction(anyString()).then(i -> i.getArgumentAt(0, String.class));

แน่นอนว่ามันไม่ได้มีประโยชน์อย่างที่AdditionalAnswersเดวิดวอลเลซแนะนำ แต่อาจมีประโยชน์หากคุณต้องการเปลี่ยนการโต้แย้ง "ในทันที"


1
สุกใส ขอบคุณ. หากการโต้เถียงเป็นlongเช่นนี้สิ่งนี้ยังสามารถใช้กับมวยได้Long.classหรือ
vikingsteve

1
.getArgumentAt (.. ) ไม่พบสำหรับฉัน แต่. getArgument (1) ทำงาน (mockito 2.6.2)
Curtis Yallop

41

ฉันมีปัญหาที่คล้ายกันมาก เป้าหมายคือการเยาะเย้ยบริการที่ยังคงมีออบเจ็กต์และสามารถคืนพวกเขาด้วยชื่อของพวกเขา บริการมีลักษณะดังนี้:

public class RoomService {
    public Room findByName(String roomName) {...}
    public void persist(Room room) {...}
}

บริการจำลองใช้แผนที่เพื่อจัดเก็บอินสแตนซ์ของห้อง

RoomService roomService = mock(RoomService.class);
final Map<String, Room> roomMap = new HashMap<String, Room>();

// mock for method persist
doAnswer(new Answer<Void>() {
    @Override
    public Void answer(InvocationOnMock invocation) throws Throwable {
        Object[] arguments = invocation.getArguments();
        if (arguments != null && arguments.length > 0 && arguments[0] != null) {
            Room room = (Room) arguments[0];
            roomMap.put(room.getName(), room);
        }
        return null;
    }
}).when(roomService).persist(any(Room.class));

// mock for method findByName
when(roomService.findByName(anyString())).thenAnswer(new Answer<Room>() {
    @Override
    public Room answer(InvocationOnMock invocation) throws Throwable {
        Object[] arguments = invocation.getArguments();
        if (arguments != null && arguments.length > 0 && arguments[0] != null) {
            String key = (String) arguments[0];
            if (roomMap.containsKey(key)) {
                return roomMap.get(key);
            }
        }
        return null;
    }
});

ตอนนี้เราสามารถทำการทดสอบของเราบนเยาะเย้ยนี้ ตัวอย่างเช่น:

String name = "room";
Room room = new Room(name);
roomService.persist(room);
assertThat(roomService.findByName(name), equalTo(room));
assertNull(roomService.findByName("none"));

34

ด้วย Java 8 คำตอบของ Steveจะกลายเป็น

public void testMyFunction() throws Exception {
    Application mock = mock(Application.class);
    when(mock.myFunction(anyString())).thenAnswer(
    invocation -> {
        Object[] args = invocation.getArguments();
        return args[0];
    });

    assertEquals("someString", mock.myFunction("someString"));
    assertEquals("anotherString", mock.myFunction("anotherString"));
}

แก้ไข: แม้แต่สั้น:

public void testMyFunction() throws Exception {
    Application mock = mock(Application.class);
    when(mock.myFunction(anyString())).thenAnswer(
        invocation -> invocation.getArgument(0));

    assertEquals("someString", mock.myFunction("someString"));
    assertEquals("anotherString", mock.myFunction("anotherString"));
}

6

นี่เป็นคำถามที่ค่อนข้างเก่า แต่ฉันคิดว่ายังคงเกี่ยวข้อง นอกจากนี้คำตอบที่ได้รับการยอมรับก็ใช้งานได้กับ String เท่านั้น ในขณะเดียวกันก็มี Mockito 2.1 และการนำเข้ามีการเปลี่ยนแปลงดังนั้นฉันต้องการแบ่งปันคำตอบปัจจุบันของฉัน:

import static org.mockito.AdditionalAnswers.returnsFirstArg;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.when;

@Mock
private MyClass myClass;

// this will return anything you pass, but it's pretty unrealistic
when(myClass.myFunction(any())).then(returnsFirstArg());
// it is more "life-like" to accept only the right type
when(myClass.myFunction(any(ClassOfArgument.class))).then(returnsFirstArg());

myClass.myFunction จะมีลักษณะดังนี้:

public class MyClass {
    public ClassOfArgument myFunction(ClassOfArgument argument){
        return argument;
    }  
}

4

ฉันใช้สิ่งที่คล้ายกัน (โดยทั่วไปเป็นแนวทางเดียวกัน) บางครั้งมันมีประโยชน์ที่จะให้วัตถุจำลองคืนเอาต์พุตที่กำหนดไว้ล่วงหน้าสำหรับอินพุตบางตัว มันจะเป็นแบบนี้:

private Hashtable<InputObject,  OutputObject> table = new Hashtable<InputObject, OutputObject>();
table.put(input1, ouput1);
table.put(input2, ouput2);

...

when(mockObject.method(any(InputObject.class))).thenAnswer(
       new Answer<OutputObject>()
       {
           @Override
           public OutputObject answer(final InvocationOnMock invocation) throws Throwable
           {
               InputObject input = (InputObject) invocation.getArguments()[0];
               if (table.containsKey(input))
               {
                   return table.get(input);
               }
               else
               {
                   return null; // alternatively, you could throw an exception
               }
           }
       }
       );

4

คุณอาจต้องการใช้ Verify () ร่วมกับ ArgumentCaptor เพื่อรับรองการดำเนินการในการทดสอบและ ArgumentCaptor เพื่อประเมินข้อโต้แย้ง:

ArgumentCaptor<String> argument = ArgumentCaptor.forClass(String.class);
verify(mock).myFunction(argument.capture());
assertEquals("the expected value here", argument.getValue());

ค่าของอาร์กิวเมนต์สามารถเข้าถึงได้อย่างชัดเจนผ่าน argument.getValue () สำหรับการจัดการ / ตรวจสอบเพิ่มเติม / อะไรก็ตาม


3

นี่เก่าไปหน่อย แต่ฉันมาที่นี่เพราะฉันมีปัญหาเดียวกัน ฉันใช้ JUnit แต่คราวนี้ในแอป Kotlin กับ mockk ฉันโพสต์ตัวอย่างที่นี่สำหรับการอ้างอิงและการเปรียบเทียบกับคู่ Java:

@Test
fun demo() {
  // mock a sample function
  val aMock: (String) -> (String) = mockk()

  // make it return the same as the argument on every invocation
  every {
    aMock.invoke(any())
  } answers {
    firstArg()
  }

  // test it
  assertEquals("senko", aMock.invoke("senko"))
  assertEquals("senko1", aMock.invoke("senko1"))
  assertNotEquals("not a senko", aMock.invoke("senko"))
}

2

คุณสามารถทำได้โดยใช้ArgumentCaptor

ลองนึกภาพคุณมีฟังก์ชั่นถั่วเช่นนั้น

public interface Application {
  public String myFunction(String abc);
}

จากนั้นในชั้นทดสอบของคุณ:

//Use ArgumentCaptor to capture the value
ArgumentCaptor<String> param = ArgumentCaptor.forClass(String.class);


when(mock.myFunction(param.capture())).thenAnswer(new Answer<String>() {
    @Override
    public String answer(InvocationOnMock invocation) throws Throwable {
      return param.getValue();//return the captured value.
    }
  });

หรือถ้าคุณชื่นชอบแลมบ์ดาคุณก็ทำได้:

//Use ArgumentCaptor to capture the value
ArgumentCaptor<String> param = ArgumentCaptor.forClass(String.class);


when(mock.myFunction(param.capture()))
    .thenAnswer((invocation) -> param.getValue());

สรุป:ใช้ argumentcaptor เพื่อดักจับพารามิเตอร์ที่ส่งผ่าน ตอบกลับมาคืนค่าที่บันทึกโดยใช้ getValue


สิ่งนี้ไม่ทำงาน (อีกต่อไป?) เกี่ยวกับเอกสาร: วิธีนี้จะต้องใช้ภายในการตรวจสอบ ซึ่งหมายความว่าคุณสามารถดักจับค่าเมื่อใช้วิธีการยืนยัน
Muhammed Misir

1. ไม่แน่ใจว่าสิ่งที่คุณหมายถึงโดยThis doesn´t work (anymore?).ฉันได้ทำงานในอินสแตนซ์ของฉันนี้ 2. ขออภัยฉันไม่ชัดเจนในจุดที่คุณพยายามจะทำ คำตอบเฉพาะสำหรับคำถามของ OP
Cyril Cherian
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.