모의는 실제 물체의 거동을 테스트하는 것을 목표로 합니다.
종속성을 시뮬레이션하므로 단위 테스트 속도를 크게 저하시킬 수 있는 외부 리소스를 호출할 필요가 없습니다.
기대를 정의하고 검증할 수 있습니다.
예를 들어 메소드가 특정 횟수 및/또는 특정 매개변수를 사용하여 호출되도록 할 수 있습니다.
use PHPUnit\Framework\TestCase; class MyTest extends TestCase { public function testMockExample(): void { $depencencyMock = $this->createMock(MyDependency::class); $dependencyMock->expects($this->exactly(2)) ->method('someMethod') ->with('some parameter'); $classToTest = new ClassToTest($dependencyMock); } }
willReturn()은 반환 유형과의 호환성을 보장합니다.
// In code class MyClass { public function getNum(): int { } } // In tests $myClassMock = $this->createMock(MyClass::class); $myClassMock->expects($this->once()) ->method('getNum') ->willReturn(2);
입력 매개변수를 기반으로 동적 동작을 테스트하려는 경우 willReturnCallback을 사용할 수도 있습니다.
모의는 실제 동작을 모방하기 때문에 요점을 놓치기 쉽습니다. 일반적인 나쁜 관행에 대해 논의해 보겠습니다.
❌ 그러지 마세요:
$colorServiceMock = $this->createMock(ColorService::class); $colorServiceMock->method('hexToName') ->willReturn('red'); $color = (new MyClass($colorServiceMock))->getColorName('ff0000');
✅ 대신에 몇 가지 기대치를 추가하세요.
$colorServiceMock->expects($this->once()) ->method('hexToName') ->with('00f00') ->willReturn('green'); $color = (new MyClass($colorServiceMock))->getColorName('00f00');
모의의 목적은 상호 작용을 확인하는 것임을 기억하세요.
SomeInterface를 구현한 MyClass를 테스트해 보겠습니다.
❌ 그러지 마세요:
$myclassMock = $this->createMock(MyClass::class);
✅ 대신 인터페이스를 모의하세요:
$myclassMock = $this->createMock(SomeInterface::class);
모의는 행동에 중점을 둡니다. 계약이 아닌 구현을 수정해야 하므로 인터페이스는 일반적으로 변경되지 않습니다.
Tomas Votruba는 이 문제를 훌륭하게 설명합니다. 과잉 모의 테스트에서 가치를 추출하는 5가지 방법
구성요소 간의 긴밀한 결합을 무시하기 쉽습니다.
$productRepositoryMock = $this->createMock(ProductRepository::class); $invoiceRepositoryMock = $this->createMock(InvoiceRepository::class); $emailServiceMock = $this->createMock(EmailService::class); $overComplexService = new OverComplexService($productRepositoryMock, $invoiceRepositoryMock, $emailServiceMock);
위의 예는 우려 사항의 분리를 무너뜨렸고, Mock은 이러한 나쁜 습관을 영속시키고 있습니다.
Mock은 강력한 도구이지만 단위 테스트만으로는 충분하지 않습니다. 그 밖에도 다양한 유형의 테스트(예: 통합, e2e)가 필요합니다.
나쁜 관행 외에도 프로젝트에서 모의 객체가 오용되거나 과도하게 사용되었음을 나타낼 수 있는 다른 징후가 있습니다.
Martin Fowler는 Mocks가 Stubs가 아닌 이유를 설명하는 환상적인 게시물을 작성했습니다.
이를 사용할 수 있는 구체적인 상황을 살펴보겠습니다.
다음은 모의 테스트가 더 적합한 몇 가지 테스트 사례입니다.
PHPUnit을 사용하면 매우 편리하게 스텁을 만들 수 있습니다.
use PHPUnit\Framework\TestCase; class MyTest extends TestCase { public function testMockExample(): void { $depencencyMock = $this->createMock(MyDependency::class); $dependencyMock->expects($this->exactly(2)) ->method('someMethod') ->with('some parameter'); $classToTest = new ClassToTest($dependencyMock); } }
다음은 스텁이 더 적합한 몇 가지 테스트 사례입니다.
간단히 말하면 스텁은 실제 객체의 동작을 확인하기 위한 것이 아니라 상태를 확인하기 위한 것입니다.
단위 테스트의 주요 목적은 각 단위/구성 요소가 예상대로 작동하는지 확인하는 것이지만, 실제 코드 외에 이러한 테스트도 유지해야 합니다.
스텁은 테스트 설정을 단순화할 수 있으며 메서드 호출 및 상호 작용을 추적할 필요가 없는 간단한 시나리오에 매우 효율적입니다.
일부 테스트에 집중하여 불필요한 복잡성을 방지할 수 있습니다.
모의는 메소드 호출과 해당 매개변수를 추적할 수 있습니다.
실제 동작을 대표하는 값을 반환하는 것을 잊지 마세요. 그렇지 않으면 잘못된 안보의식을 갖게 될 수도 있습니다.
Mock은 유지 관리에 불필요한 복잡성을 피하기 위해 자제해서 사용해야 합니다.
위 내용은 PHP: 조롱해야 할까요, 아니면 가야 할까요?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!