Почему ядро ​​.net побуждает вас создавать интерфейсы для всех ваших зависимостей? - PullRequest
0 голосов
/ 22 февраля 2019

Я читаю документацию Microsoft по внедрению зависимостей и продолжаю видеть один и тот же шаблон.Автор определит какой-либо вид службы, создаст интерфейс для службы, а затем добавит интерфейс в коллекцию служб в качестве инъекционной зависимости.

public class MyService : IMyService {
}

_

public interface IMyService {
}

Затемпри запуске, в вашем методе ConfigureServices, вы регистрируете свою службу следующим образом

services.AddScoped<IMyService>();

Затем в контроллере или другом классе вы вводите интерфейс как зависимость

public class SampleDataController : Controller {

    private readonly IMyService _service;

    public SampleDataController(IMyService service) {
       _service = service;
    }

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

1 Ответ

0 голосов
/ 22 февраля 2019

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

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

public class FooService
{
    private readonly SqlConnection _db;

    public FooService(SqlConnection db)
    {
        _db = db;
    }
}

Из-за жесткой зависимости SqlConnection здесь вы можете использовать только этого конкретного поставщика.При модульном тестировании этой службы вам необходимо будет подключиться к реальному экземпляру SQL Server, поскольку нет способа абстрагировать эту зависимость.Затем у вас есть дополнительная переменная, поскольку ваш тест может не пройти из-за реальной проблемы с вашим сервисом или просто из-за проблем с подключением к SQL Server.Вы никогда не узнаете.

Теперь представьте, что вы сделали это вместо этого:

public class FooService
{
    private readonly IDbConnection _db;

    public FooService(IDbConnection db)
    {
        _db = db;
    }
}

Теперь вы можете передать все, что реализует этот интерфейс.Вы можете использовать SqlConnection, OracleConnection и т. Д. Или даже полностью смоделированный IDbConnection, который хранит данные в памяти и возвращает готовые ответы.

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