依存性注入のプロトコル構成

合成と依存関係の注入を使用したいのですが、各エンティティにいくつかの依存性が注入され始めると、特定の積み上げが得られます。


プロジェクトは成長しており、オブジェクトにますます多くの依存関係を注入し、メソッドを何度もリファクタリングする必要があります。


しかし、より管理しやすい方法があります。



この記事は、DIの使用を開始したばかりのユーザー、またはまったく使用していないユーザーで、IoCに特に慣れていないユーザーに役立ちます。 このアプローチは他の言語にも適用されますが、作成者がSwiftで記述しているため、すべての例がSwiftにあります(約Per。)


問題


突然ImageProviderプロバイダーを必要とするオブジェクトがある場合、次のように記述します。


 class FooCoordinator { let imageProvider: ImageProvider init(..., imageProvider: ImageProvider) ///... } 

非常にシンプルで便利+これにより、テストでプロバイダーを置き換えることができます。


コードベースが大きくなるにつれて、依存性がますます現れ、それぞれが以下を強制します。



たとえば、数か月後、オブジェクトには3つの依存関係がある場合があります。


 class FooCoordinator { let imageProvider: ImageProvider let articleProvider: ArticleProvider let persistanceProvider: PersistanceProvider init(..., imageProvider: ImageProvider, articleProvider: ArticleProvider, persistanceProvider: PersistanceProvider) { self.imageProvider = imageProvider self.articleProvider = articleProvider self.persistanceProvider = persistanceProvider ///... } ///... } 

通常、プロジェクトには複数のクラスがあるため、同じパターンが何度も繰り返されます。


そして、たとえばAppControllerFlow Coordinatorなど、これらすべての依存関係へのリンクを保存する場所が必要であることを忘れてはなりません。


このアプローチは必然的に痛みにつながります。 痛みは、シングルトンなど、あまり適切でない決定の使用を動機付けることができます。


しかし、コードインジェクションの利点をすべて備えた、シンプルで簡単なコードサポートが必要です。


代替案


プロトコルの構成(インターフェイス)を使用して、コードの可読性とサービス品質を向上させることができます。


依存関係のベースプロトコルコンテナについて説明しましょう。


 protocol Has{Dependency} { var {dependency}: {Dependency} { get } } 

{Dependency}をオブジェクトの名前に変更します


たとえば、 ImageProviderようになります。


 protocol HasImageProvider { var imageProvider: ImageProvider { get } } 

Swiftでは、 &演算子を使用してプロトコルからコンポジションを作成できます。つまり、エンティティに依存関係のリポジトリを1つだけ含めることができます。


 class FooCoordinator { typealias Dependencies = HasImageProvider & HasArticleProvider let dependencies: Dependencies init(..., dependencies: Dependencies) } 

AppControllerまたはFlow Coordinator (またはエンティティの作成に使用されるもの)で、1つのコンテナの下にあるすべての依存関係を構造の形で非表示にできます。


 struct AppDependency: HasImageProvider, HasArticleProvider, HasPersistanceProvider { let imageProvider: ImageProvider let articleProvider: ArticlesProvider let persistanceProvider: PersistenceProvider } 

そして、すべてのアプリケーションの依存関係は、ロジックや魔法のようなものではなく、単なる構造を持つコンテナに保存されます。


このアプローチにより、読みやすさが向上し、すべての依存関係が一緒に保存されますが、さらに重要なことは、オブジェクトに必要な依存関係に関係なく、構成コードは常に同じであるということです。


 class FlowCoordinator { let dependencies: AppDependency func configureViewController(vc: ViewController) { vc.dependencies = dependencies } } 

各オブジェクトは、実際に必要な依存関係のみを記述し、それらのみを受け取ります。


たとえば、 FooCoordinatorImageProvider必要とする場合ImageProvider AppDependency構造をImageProvider Swiftの型チェックはImageProviderへのアクセスのみを提供します


将来的に、たとえばPersistanceProviderような依存関係がさらに必要になった場合、 typealiasに追加するだけです。


 class FooCoordinator { typealias Dependencies = HasImageProvider & HasArticleProvider & HasPersistanceProvider } 

それだけです


このアプローチにはいくつかの利点があります。




Source: https://habr.com/ru/post/J326712/


All Articles