Впровадження залежностей
Впровадження залежності (англ. Dependency injection, DI) — шаблон проєктування програмного забезпечення, що передбачає надання зовнішньої залежності програмному компоненту, використовуючи «інверсію управління» (англ. Inversion of control, IoC) для розв'язання (отримання) залежностей.
Впровадження — це передача залежності (тобто, сервісу) залежному об'єкту (тобто, клієнту). Передавати залежності клієнту замість дозволити клієнту створити сервіс є фундаментальною вимогою до цього шаблону проєктування.
Існує три найбільш поширені форми впровадження залежностей:
- впровадження в конструктор,
- впровадження у властивість,
- впровадження в метод.
Огляд
Впровадження залежностей — це шаблон проєктування, в якому залежності (або сервіси) впроваджуються, або передаються по посиланню в залежний об'єкт (клієнт) і стають частиною клієнтського стану. Шаблон відокремлює створення залежностей клієнта від власної логіки клієнта, що дозволяє компонентам бути слабко зв'язаними і притримуватися принципів інверсії залежностей і єдиного обов'язку. Це суперечить анти-шаблону service locator, який дозволяє клієнтам знати про систему, що використовується для пошуку залежностей.
Переваги
- Оскільки впровадження залежностей не вимагає змін у поведінці коду, його можна застосувати як рефакторинг. В результаті цього клієнти стають більш незалежними і над ними легше проводити модульне тестування в ізоляції з використанням макетів об'єкта, які імітують інші об'єкти, від яких залежить об'єкт, що тестується. Простота тестування найчастіше є першою помітною перевагою використання впровадження залежностей.
- Впровадження залежностей не вимагає від клієнта знань про конкретну реалізацію, яку йому потрібно використовувати. Це дозволяє ізолювати клієнт від впливу змін проєктування і дефектів. Це сприяє повторному використанню, тестуванню і підтримці коду.
- Впровадження залежностей може використовуватися для перенесення деталей конфігурації системи в конфігураційні файли, що дозволяє системі змінювати конфігурацію без перекомпіляції. Окремі конфігурації можуть бути написані для різних ситуацій, що вимагають різних реалізацій компонентів.
- Впровадження залежностей сприяє паралельній і незалежній розробці. Два розробника можуть незалежно створювати класи, які використовують один одного, знаючи тільки про інтерфейси, через які класи співпрацюють.
- Впровадження залежностей знижує зв'язність між класом і його залежностями.
Структура
UML класи та діаграма послідовності
В наведеній вище UML діаграмі класів, клас Client
, який потребує об'єкти ServiceA
та ServiceB
не інстанціює класи ServiceA1
та ServiceB1
напряму. Замість цього клас Injector
створює об'єкти та впроваджує їх в Client
, що робить Client
незалежним від того як, ці об'єкти створюються (який конкретно клас інстанціюється).
UML діаграма послідовності демонструє взаємодії часу виконання: Об'єкт Injector
створює об'єкти ServiceA1
та ServiceB1
. Після цього Injector
створює об'єкт Client
і впроваджує в нього об'єкти ServiceA1
та ServiceB1
.
Приклади
Без впровадження залежностей
В наступному C# прикладі, клас Client
містить в собі поле класу Service
, яке ініціалізується в конструкторі класу Client
. Клієнт має контроль над тим, яка реалізація сервіса використовується, оскільки сам створює її. В цьому прикладі клієнт має жорстку залежність від ServiceExample()
.
// An example without dependency injection
public class Client
{
// Internal reference to the service used by this client
private readonly Service _service;
// Constructor
public Client()
{
// Specify a specific implementation in the constructor instead of using dependency injection
_service = new ServiceExample();
}
// Method within this client that uses the services
public string Greet()
{
return "Hello " + _service.Name;
}
}
Впровадження через конструктор
Клас, якому потрібна залежність, повинен надати відкритий конструктор, який приймає екземпляр необхідної залежності як аргумент конструктора. У більшості випадків, це повинен бути тільки public конструктор. Якщо необхідна більш ніж одна залежність, можуть бути використані додаткові аргументи конструктора. Є найбільш широко вживаним і рекомендованим методом впровадження залежностей.
public class Client
{
private readonly IService _service;
// Constructor
public Client(IService service)
{
if (service == null)
{
throw new ArgumentNullException("service");
}
_service = service;
}
// Method within this client that uses the services
public string Greet()
{
return "Hello " + _service.Name;
}
}
Впровадження через властивість
Клас, який використовує залежність, повинен надати відкриту, доступну для запису властивість типу залежності. Client
залежить від IService
. Клієнти можуть поставляти реалізації IService
, встановлюючи властивість Dependency
. На відміну від впровадження в конструктор, ви не можете відзначити поле властивості Dependency
як readonly
, тому що ви дозволяєте викликаючим елементам змінювати цю властивість в будь-який момент життєвого циклу Client
. Однак така реалізація є крихкою, тому що немає гарантії, що властивість Dependency повертає екземпляр IService. Код, як цей, викине NullReferenceException, якщо значення властивості Dependency — null.
public class Client
{
public IService Dependency {get; set;}
// Method within this client that uses the services
public string Greet()
{
return "Hello " + Dependency.Name;
}
}
Впровадження через метод
Елемент, що викликає метод, впроваджує залежність як параметр методу в кожен виклик методу. Впровадження в метод краще використовувати тоді, коли залежність може змінюватися з кожним викликом методу. Це може бути в тому випадку, коли залежність сама по собі представляє значення, або коли елемент, що викликає, надає споживачеві інформацію про контекст, в якому викликається операція.
public class Client
{
// Method within this client that uses the services
public string Greet(IService service)
{
if (service == null)
{
throw new ArgumentNullException("service");
}
return "Hello " + service.Name;
}
}
Конфігурування через XML
Конфігурування DI-контейнеру Unity за допомогою XML
<register type="IBasketService" mapTo="BasketService" />
<register type="BasketDiscountPolicy" mapTo="RepositoryBasketDiscountPolicy" />
<register type="BasketRepository" mapTo="SqlBasketRepository">
<constructor>
<param name="connString">
<value value="CommerceObjectContext" typeConverter="ConnectionStringConverter" />
</param>
</constructor>
</register>
<register type="DiscountRepository" mapTo="SqlDiscountRepository">
<constructor>
<param name="connString">
<value value="CommerceObjectContext" typeConverter="ConnectionStringConverter" />
</param>
</constructor>
</register>
<register type="ProductRepository" mapTo="SqlProductRepository">
<constructor>
<param name="connString">
<value value="CommerceObjectContext" typeConverter="ConnectionStringConverter" />
</param>
</constructor>
</register>
<register type="CurrencyProvider" mapTo="SqlCurrencyProvider">
<constructor>
<param name="connString">
<value value="CommerceObjectContext" typeConverter="ConnectionStringConverter" />
</param>
</constructor>
</register>
Перетворення інтерфейсу IBasketService в клас BasketService реалізується за допомогою простого елемента register. Деякі конкретні класи приймають рядок з'єднання як вхідні дані, тому необхідно визначити, яким чином знаходиться значення цього рядка. Що стосується Unity, можна зробити це, вказавши, що ви використовуєте користувацький тип конвертера під назвою ConnectionStringConverter. Цей конвертер буде шукати значення CommerceObjectContext серед стандартних рядків з'єднання web.config і повертати рядок з'єднання з цим ім'ям. Решта елементів повторюють ці два патерни. Оскільки Unity може автоматично перетворювати запити в конкретні типи, навіть якщо відсутні явні реєстрації, вам не потрібно застосовувати XML-елементи для HomeController і BasketController. Завантаження конфігурації в контейнер виконується за допомогою виклику єдиного методу:
container.LoadConfiguration();
Попередження
Як тільки ваш застосунок(ісп.) буде виростати в розмірах і ускладнюватися, теж саме буде відбуватися і з вашим конфігураційним файлом, якщо ви використовуєте конфігураційну композицію. Він може стати справжньою проблемою, оскільки цей файл моделює такі сутності коду, як класи, параметри тощо, але без переваг компілятора, опцій налагодження і т. д. Файли будуть ставати крихкими і непрозорими з точки зору наявності помилок, тому використовуйте даний підхід тільки, якщо вам необхідно пізнє зв'язування.
Конфігурування програми за допомогою коду
c.For<IBasketService>().Use<BasketService>();
c.For<BasketDiscountPolicy>().Use<RepositoryBasketDiscountPolicy>();
string connectionString = ConfigurationManager.ConnectionStrings["CommerceObjectContext"].ConnectionString;
c.For<BasketRepository>().Use<SqlBasketRepository>().Ctor<string>().Is(connectionString);
c.For<DiscountRepository>().Use<SqlDiscountRepository>().Ctor<string>().Is(connectionString);
c.For<ProductRepository>().Use<SqlProductRepository>().Ctor<string>().Is(connectionString);
c.For<CurrencyProvider>().Use<SqlCurrencyProvider>().Ctor<string>().Is(connectionString);
Для того щоб підтримати ті класи, для яких потрібен рядок з'єднання, ви продовжуєте послідовність For/Use шляхом виклику методу Ctor та передачі рядка з'єднання. Метод Ctor виконує пошук строкового параметра в конструкторі конкретного класу і використовує передане значення для цього параметра.
Використання коду як конфігурації не тільки компактніше XML-конфігурації, але також підтримується компілятором. Типи аргументів являють собою реальні типи, які перевіряє компілятор. Змінна API StructureMap поставляється навіть з деякими видовими обмежувачами, які повідомляють компілятору про перевірку того, чи збігається тип, який визначається методом використання з абстракціями, позначеними за допомогою методу For. Якщо перетворення неможливо, то код не компілюється.
Незважаючи на те, що технологія використання коду як конфігурації безпечна і проста в застосуванні, її потрібно більше супроводжувати. Щоразу при додаванні у застосунок нового типу ви також повинні пам'ятати і про його реєстрацію.
Див. також
Посилання
- Dependency Injection in .NET — Mark Seemann, Manning, 2011
- Inversion of Control Containers and the Dependency Injection pattern — Martin Fowler.
- Design Patterns: Dependency Injection — MSDN Magazine, September 2005