Внедрение зависимостей через конструкторы или установщики свойств?

Я занимаюсь рефакторингом класса и добавляю в него новую зависимость. Класс в настоящее время принимает свои существующие зависимости в конструкторе. Поэтому для согласованности я добавляю параметр в конструктор.
Конечно, есть несколько подклассов плюс еще больше для юнит-тестов, так что теперь я играю в игру об изменении всех конструкторов, чтобы соответствовать, и это требует времени.
Это заставляет меня думать, что использование свойств с установщиками - лучший способ получения зависимостей. Я не думаю, что внедренные зависимости должны быть частью интерфейса для создания экземпляра класса. Вы добавляете зависимость, и теперь все ваши пользователи (подклассы и все, кто создает для вас экземпляр) внезапно узнают об этом. Это похоже на разрыв инкапсуляции.

Это не похоже на шаблон с существующим здесь кодом, поэтому я хочу выяснить, каково общее согласие, плюсы и минусы конструкторов в сравнении со свойствами. Лучше ли использовать установщики свойств?

15 ответов

Решение

Смотря как:-).

Если класс не может выполнять свою работу без зависимости, то добавьте ее в конструктор. Классу нужна новая зависимость, поэтому вы хотите, чтобы ваши изменения сломали вещи. Кроме того, создание класса, который не полностью инициализирован ("двухэтапное построение"), является анти-паттерном (IMHO).

Если класс может работать без зависимости, установщик в порядке.

Предполагается, что пользователи класса знают о зависимостях данного класса. Если бы у меня был класс, который, например, подключался к базе данных и не предоставлял средства для внедрения зависимости уровня постоянства, пользователь никогда бы не узнал, что соединение с базой данных должно быть доступно. Однако, если я изменю конструктор, я сообщу пользователям, что существует зависимость от уровня персистентности.

Кроме того, чтобы избежать необходимости изменять каждое использование старого конструктора, просто примените цепочку конструктора в качестве временного моста между старым и новым конструктором.

public class ClassExample
{
    public ClassExample(IDependencyOne dependencyOne, IDependencyTwo dependencyTwo)
        : this (dependnecyOne, dependencyTwo, new DependnecyThreeConcreteImpl())
    { }

    public ClassExample(IDependencyOne dependencyOne, IDependencyTwo dependencyTwo, IDependencyThree dependencyThree)
    {
        // Set the properties here.
    }
}

Одной из точек внедрения зависимостей является выявление зависимостей класса. Если у класса слишком много зависимостей, возможно, пора провести рефакторинг: использует ли каждый метод класса все зависимости? Если нет, то это хорошая отправная точка, чтобы увидеть, где класс может быть разделен.

Конечно, добавление конструктора означает, что вы можете проверить все сразу. Если вы назначаете вещи в поля только для чтения, то у вас есть некоторые гарантии относительно зависимостей вашего объекта прямо со времени создания.

Добавление новых зависимостей - это настоящая боль, но, по крайней мере, таким образом компилятор продолжает жаловаться, пока не станет правильным. Думаю, это хорошо.

Общий предпочтительный подход состоит в том, чтобы максимально использовать инжектор конструктора.

Внедрение в конструктор точно определяет, каковы необходимые зависимости для правильного функционирования объекта - нет ничего более раздражающего, чем обновление объекта и его сбой при вызове метода, поскольку некоторая зависимость не установлена. Объект, возвращаемый конструктором, должен находиться в рабочем состоянии.

Старайтесь иметь только один конструктор, он сохраняет дизайн простым и позволяет избежать двусмысленности (если не для людей, для контейнера DI).

Вы можете использовать инъекцию свойств, когда у вас есть то, что Марк Симанн называет локальным значением по умолчанию в своей книге "Инъекция зависимостей в.NET": зависимость является необязательной, потому что вы можете обеспечить хорошо работающую реализацию, но хотите позволить вызывающей стороне указать другую, если необходимо.

(Бывший ответ ниже)


Я думаю, что инъекция в конструктор лучше, если инъекция обязательна. Если это добавляет слишком много конструкторов, рассмотрите возможность использования фабрик вместо конструкторов.

Инъекция сеттера хороша, если инъекция не обязательна, или если вы хотите изменить ее на полпути. Я вообще не люблю сеттеров, но это дело вкуса.

Если у вас есть большое количество необязательных зависимостей (которые уже пахнут), то, вероятно, инъекция сеттера - путь. Внедрение в конструктор лучше показывает ваши зависимости.

Это во многом вопрос личного вкуса. Лично я предпочитаю инъекцию сеттера, потому что считаю, что это дает вам больше гибкости в способе замены реализаций во время выполнения. Кроме того, конструкторы с большим количеством аргументов, на мой взгляд, не являются чистыми, и аргументы, предоставляемые в конструкторе, должны быть ограничены необязательными аргументами.

Пока интерфейс классов (API) понятен в том, что ему нужно для выполнения своей задачи, у вас все хорошо.

Я лично предпочитаю "шаблон" " Извлечь и переопределить ", а не вводить зависимости в конструктор, в основном по причине, изложенной в вашем вопросе. Вы можете установить свойства как virtual а затем переопределить реализацию в производном тестируемом классе.

Я предпочитаю внедрение конструктора, потому что оно помогает "обеспечить" требования к зависимости класса. Если это в c'or, потребитель должен установить объекты, чтобы приложение компилировалось. Если вы используете сеттерную инъекцию, они могут не знать, что у них есть проблема, до времени выполнения - и в зависимости от объекта это может быть слишком поздно во время выполнения.

Время от времени я все еще использую инъекцию сеттера, когда внедренному объекту, возможно, понадобится куча работы, например, инициализация.

Я предпочитаю внедрение конструктора, потому что это кажется наиболее логичным. Это все равно что сказать, что мой класс требует, чтобы эти зависимости выполняли свою работу. Если это необязательная зависимость, то свойства кажутся разумными.

Я также использую инъекцию свойств для установки вещей, на которые у контейнера нет ссылок, таких как представление ASP.NET на презентере, созданном с использованием контейнера.

Я не думаю, что это нарушает инкапсуляцию. Внутренняя работа должна оставаться внутренней, а зависимости связаны с другой проблемой.

Недавно я столкнулся с ситуацией, когда у меня было несколько зависимостей в классе, но в каждой реализации обязательно менялась только одна из этих зависимостей. Поскольку зависимости доступа к данным и регистрации ошибок, вероятно, будут изменены только для целей тестирования, я добавил необязательные параметры для этих зависимостей и предоставил реализации этих зависимостей по умолчанию в своем коде конструктора. Таким образом, класс сохраняет свое поведение по умолчанию, если только он не переопределен потребителем класса.

Использование необязательных параметров может быть выполнено только в платформах, которые их поддерживают, таких как.NET 4 (как для C#, так и для VB.NET, хотя в VB.NET они всегда были). Конечно, вы можете реализовать аналогичную функциональность, просто используя свойство, которое может быть переназначено потребителем вашего класса, но вы не получаете преимущества неизменности, обеспечиваемого наличием объекта частного интерфейса, назначенного параметру конструктора.

Все это, как говорится, если вы вводите новую зависимость, которая должна предоставляться каждым потребителем, вам придется реорганизовать ваш конструктор и весь код, который потребляет ваш класс. Мои предложения, приведенные выше, действительно применимы только в том случае, если вы можете позволить себе реализацию по умолчанию для всего вашего текущего кода, но при этом можете при необходимости переопределить реализацию по умолчанию.

Один из вариантов, который стоит рассмотреть, - составление сложных множественных зависимостей из простых одиночных зависимостей. То есть определить дополнительные классы для составных зависимостей. Это немного упрощает внедрение конструктора WRT - меньшее количество параметров на вызов - и в то же время поддерживает необходимость обязательного предоставления всех зависимостей для создания экземпляра.

Конечно, это имеет смысл, если есть какая-то логическая группировка зависимостей, так что соединение представляет собой нечто большее, чем произвольный агрегат, и это имеет смысл, если существует несколько зависимостей для одной составной зависимости, но блок параметров "pattern" имеет был вокруг в течение долгого времени, и большинство из тех, что я видел, были довольно произвольны.

Лично я больше фанат использования методов / установщиков свойств для определения зависимостей, опций и т. Д. Имена вызовов помогают описать происходящее. Хорошей идеей будет привести примеры фрагментов "это как установить" и убедиться, что зависимый класс выполняет достаточное количество проверок на ошибки. Возможно, вы захотите использовать модель конечного состояния для установки.

Внедрение в конструктор явно выявляет зависимости, делая код более читабельным и менее подверженным необработанным ошибкам во время выполнения, если аргументы проверяются в конструкторе, но это действительно сводится к личному мнению, и чем больше вы используете DI, тем больше вы будете использовать имеют тенденцию качаться взад и вперед так или иначе в зависимости от проекта. У меня лично есть проблемы с запахами кода как конструкторы с длинным списком аргументов, и я чувствую, что потребитель объекта должен знать зависимости, чтобы в любом случае использовать объект, так что это оправдывает использование инъекции свойства. Мне не нравится неявная природа внедрения свойств, но я нахожу это более элегантным, в результате чего код выглядит чище. Но, с другой стороны, внедрение в конструктор действительно обеспечивает более высокую степень инкапсуляции, и по своему опыту я стараюсь избегать конструкторов по умолчанию, поскольку они могут негативно повлиять на целостность инкапсулированных данных, если не соблюдать осторожность.

Выбирайте инъекцию по конструктору или по свойствам в зависимости от вашего конкретного сценария. И не думайте, что вы должны использовать DI только потому, что это кажется необходимым, и это предотвратит плохой дизайн и запах кода. Иногда не стоит использовать шаблон, если усилия и сложность перевешивают выгоду. Будь проще.

Это старый пост, но если он понадобится в будущем, возможно, он будет полезен:

https://github.com/omegamit6zeichen/prinject

У меня была похожая идея и я придумал эту структуру. Это, вероятно, далеко от завершения, но это идея структуры, ориентированной на внедрение свойств

Когда использовать внедрение конструктора?
Когда мы хотим убедиться, что объект создан со всеми его зависимостями, и убедиться, что требуемые зависимости не равны нулю.

Когда использовать сеттер-инъекцию?
Когда мы работаем с необязательными зависимостями, которым можно присвоить разумные значения по умолчанию внутри класса. В противном случае проверки на ненулевое значение должны выполняться везде, где код использует зависимость. Кроме того, методы установки делают объекты этого класса открытыми для реконфигурации или повторной инъекции в более позднее время.

Источники:документация Spring ,Java Revisited.

Это зависит от того, как вы хотите реализовать. Я предпочитаю внедрение в конструктор везде, где я чувствую, что значения, которые входят в реализацию, меняются не часто. Например: если стратегия compnay идет с сервером oracle, я настрою значения моего источника данных для соединений bean-компонентов, получающих соединения через внедрение конструктора. В противном случае, если мое приложение является продуктом и, скорее всего, оно может подключиться к любой базе данных клиента, я бы реализовал такую ​​конфигурацию базы данных и мультибрендовую реализацию посредством внедрения сеттера. Я только что взял пример, но есть более эффективные способы реализации сценариев, которые я упомянул выше.

Другие вопросы по тегам