検索
ホームページバックエンド開発PHPチュートリアルPHP: 嘲笑すべきですか、それともやめるべきですか?

PHP: Should I mock or should I go?

要するにモック

モックは、実際のオブジェクトの動作をテストすることを目的としています。

依存関係をシミュレートするため、単体テストの速度が大幅に低下する可能性がある外部リソースを呼び出す必要はありません。

期待を定義し、それを検証できます。

たとえば、メソッドが特定の回数呼び出されたり、特定のパラメータを使用して呼び出されたりするようにすることができます。

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);

上記の例は関心事の分離を破壊しており、モックはその悪い習慣を永続させています。

モックのみに依存する

モックは強力なツールですが、単体テストだけでは十分ではありません。他にもさまざまなタイプのテスト (統合、e2e など) が必要です。

モックの不適切な使用を見分ける方法

悪い習慣に加えて、プロジェクト内でモックが誤用または過剰に使用されていることを示す兆候が他にもあります。

  • テストは現実世界のシナリオを反映しておらず、本番環境の重要な問題を見落としています
  • テストと実装の間に密接な関係があるため、関連するモックが頻繁に更新されます
  • テストが複雑すぎるため、読み取りや保守が難しくなります

モックとスタブ

Martin Fowler は、モックがスタブではない理由を説明する素晴らしい投稿を書きました。

これらを使用できる具体的な状況を見てみましょう:

モックを使用する場合

モックがより意味のあるテストケースをいくつか示します:

  • クラスがその依存関係とどのように対話するかをテストする必要があります
  • 特定のメソッドが異なるパラメータで複数回呼び出される複雑なシーケンスをチェックする必要があります

スタブを使用する場合

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);
   }
}

スタブがより意味のあるテストケースをいくつか示します:

  • 相互作用を検証することなく、コードの出力または状態をテストしたい
  • 実際のデータベースを操作せずにいくつかの計算をテストする必要があります

一言で言えば、スタブは実際のオブジェクトの動作をチェックすることを目的としたものではなく、状態をチェックすることを目的としています。

微調整

単体テストの主な目的は、各単体/コンポーネントが期待どおりに動作することを確認することですが、実際のコードに加えてそれらのテストも保守する必要があります。

スタブはテストのセットアップを簡素化し、メソッド呼び出しや対話を追跡する必要がない単純なシナリオでは非常に効率的です。

一部のテストに焦点を当てておくことで、不必要な複雑さを防ぐことができます。

まとめ

モックはメソッド呼び出しとそのパラメータを追跡できます。

実際の動作を表す値を返すことを忘れないでください。そうしないと、誤った安心感を抱いてしまう可能性があります。

メンテナンスに不必要な複雑さを避けるために、モックは控えめに使用する必要があります。

以上がPHP: 嘲笑すべきですか、それともやめるべきですか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
PHP依存性噴射コンテナ:クイックスタートPHP依存性噴射コンテナ:クイックスタートMay 13, 2025 am 12:11 AM

aphpDependencyInjectionContaineriSATOULTAINATINAGECLASSDEPTINCIES、強化測定性、テスト可能性、および維持可能性。

PHPの依存噴射対サービスロケーターPHPの依存噴射対サービスロケーターMay 13, 2025 am 12:10 AM

SELECT DEPENTENCINGINOFCENT(DI)大規模なアプリケーションの場合、ServicElocatorは小さなプロジェクトまたはプロトタイプに適しています。 1)DIは、コンストラクターインジェクションを通じてコードのテスト可能性とモジュール性を改善します。 2)ServiceLocatorは、センター登録を通じてサービスを取得します。これは便利ですが、コードカップリングの増加につながる可能性があります。

PHPパフォーマンス最適化戦略。PHPパフォーマンス最適化戦略。May 13, 2025 am 12:06 AM

phpapplicationscanbeoptimizedforspeedandEfficiencyby:1)enabingopcacheinphp.ini、2)PreparedStatementswithpordatabasequeriesを使用して、3)LoopswithArray_filterandarray_mapfordataprocessing、4)の構成ngincasaSearverseproxy、5)

PHPメールの検証:電子メールが正しく送信されるようにしますPHPメールの検証:電子メールが正しく送信されるようにしますMay 13, 2025 am 12:06 AM

PHPemailvalidationinvolvesthreesteps:1)Formatvalidationusingregularexpressionstochecktheemailformat;2)DNSvalidationtoensurethedomainhasavalidMXrecord;3)SMTPvalidation,themostthoroughmethod,whichchecksifthemailboxexistsbyconnectingtotheSMTPserver.Impl

PHPアプリケーションをより速くする方法PHPアプリケーションをより速くする方法May 12, 2025 am 12:12 AM

tomakephpapplicationsfaster、followthesesteps:1)useopcodecachinglikeopcacheTostoredscriptbytecode.2)最小化abasequeriesecachingingindexing.3)leveragephp7機能forbettercodeefficiency.4)

PHP依存性インジェクション:コードのテスト可能性を改善しますPHP依存性インジェクション:コードのテスト可能性を改善しますMay 12, 2025 am 12:03 AM

依存性注入(DI)は、明示的に推移的な依存関係によりPHPコードのテスト可能性を大幅に改善します。 1)DI分離クラスと特定の実装により、テストとメンテナンスが柔軟になります。 2)3つのタイプのうち、コンストラクターは、状態を一貫性に保つために明示的な式依存性を注入します。 3)DIコンテナを使用して複雑な依存関係を管理し、コードの品質と開発効率を向上させます。

PHPパフォーマンスの最適化:データベースクエリの最適化PHPパフォーマンスの最適化:データベースクエリの最適化May 12, 2025 am 12:02 AM

DatabaseQueryoptimizationInpholvesseveralstrategESTOEnhancePerformance.1)selectonlynlynlyndorycolumnStoredatedataTransfer.2)useindexingtospeedupdataretrieval.3)revenmecrycachingtostoreres sultsoffrequent queries.4)

See all articles

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

Video Face Swap

Video Face Swap

完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

ホットツール

ZendStudio 13.5.1 Mac

ZendStudio 13.5.1 Mac

強力な PHP 統合開発環境

SecLists

SecLists

SecLists は、セキュリティ テスターの究極の相棒です。これは、セキュリティ評価中に頻繁に使用されるさまざまな種類のリストを 1 か所にまとめたものです。 SecLists は、セキュリティ テスターが必要とする可能性のあるすべてのリストを便利に提供することで、セキュリティ テストをより効率的かつ生産的にするのに役立ちます。リストの種類には、ユーザー名、パスワード、URL、ファジング ペイロード、機密データ パターン、Web シェルなどが含まれます。テスターはこのリポジトリを新しいテスト マシンにプルするだけで、必要なあらゆる種類のリストにアクセスできるようになります。

Dreamweaver Mac版

Dreamweaver Mac版

ビジュアル Web 開発ツール

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい