Внедрение зависимостей для универсального интерфейса двух типов - PullRequest
0 голосов
/ 08 июня 2018

Я использую универсальный интерфейс двух типов, который называется IRepository<Entity, DTO>.Как я могу инициализировать в RepositoryFactory с помощью Dependency Injection.

Интерфейс , который я пытаюсь инициализировать.

public interface IRepository<Entity, DTO> where Entity : BaseEntity where DTO : BaseDTO
{
    bool Save(Entity Entity);
    bool Save(Entity Entity, out int OutID);

    bool Update(Entity Entity);
    bool Update(Entity Entity, out int OutID);

    DTO GetRecord(Entity ID);
    DTO GetRecord(Expression<Func<Entity, bool>> predicate);

    List<DTO> GetList();
    List<DTO> GetByQuery(Expression<Func<Entity, bool>> predicate);
}

BaseEntity, BaseDTO

public abstract class BaseEntity
{
    public int ID { get; protected set; }
}

public class BaseDTO
{
    public int ID { get; protected set; }
}

AdministratorRepository

public class AdministratorRepository : Repository<AdministratorEntity, AdministratorDTO>
{
    public AdministratorRepository(string ConnectionString) : base(ConnectionString)
    {

    }

    // Implemented functions for Repository base class
}

Репозиторий базовый класс репозитория

public abstract class Repository<Entity, DTO> : IRepository<Entity, DTO> 
where Entity : BaseEntity 
where DTO : BaseDTO
{
    // Implemented functions for IRepository Interface
}

RepositoryFactory

public class RepositoryFactory<Entity, DTO> where Entity : BaseEntity where DTO : BaseDTO
{
    private static string CONNECTION_STRING = ConfigurationManager.ConnectionStrings["ConnectionString"].ConnectionString;

    public static IRepository<Entity, DTO> Create(Repositories repository)
    {
        IRepository<Entity, DTO> iRepo = null;

        switch (repository)
        {
            //  If I do this, program wants me to cast
            case Repositories.Administrator:
                //   Casting error occured in this line.
                //   AdministratorRepository derives from Repository.
                /*
                  Cannot implicitly convert type 
                 'BusinessRules.Repositories.AdministratorRepository' to 
'BusinessRules.Repositories.IRepositories.IRepository<Entity,DTO>'. 
                  An explicit conversion exists (are you missing a cast?)
                */
                iRepo = new AdministratorRepository(CONNECTION_STRING);
                break;

           // If I do this, program throws an exception
           // Exception: TypeInitializationException
           // Exception Description: The type initializer for 'BusinessRules.Repositories.RepositoryFactory`2' threw an exception.
           case Repo.Administrator:
                iRepo = (IRepository<Entity, DTO>)(new AdministratorRepository(CONNECTION_STRING));
                break;
        }

        return iRepo;
    }
}

1 Ответ

0 голосов
/ 09 июня 2018

Сценарий, который вы упомянули, - это «где вы пытаетесь назначить объект более производного типа, чем параметру универсального типа интерфейса».Чтобы разрешить назначение, вы должны определить общий интерфейс с типом параметра out.Прочитайте эту статью, чтобы понять дисперсию в C #.

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

Поэтому определите интерфейс IRepository как

public interface IRepository<out Entity, out DTO> where Entity : BaseEntity where DTO : BaseDTO
{
    //......
}

Полный код (пример приложения - проверьте это live-скрипка)

using System;

public abstract class BaseEntity
{
    public int ID { get; protected set; }
}

public class BaseDTO
{
    public int ID { get; protected set; }
}

public interface IRepository<out Entity, out DTO> where Entity : BaseEntity where DTO : BaseDTO
{
    string GetTestString();   
}

public abstract class Repository<Entity, DTO> : IRepository<Entity, DTO> 
where Entity : BaseEntity 
where DTO : BaseDTO
{

    // Implemented functions for IRepository Interface
    public Repository(string connectionString)
    {

    }

    public abstract string GetTestString();
}

public class AdministratorEntity : BaseEntity
{

}

public class AdministratorDTO : BaseDTO
{

}

public class AdministratorRepository : Repository<AdministratorEntity, AdministratorDTO>
{
    public AdministratorRepository(string ConnectionString) : base(ConnectionString)
    {

    }

    public override string GetTestString()
    {
        return "TestString";
    }

    // Implemented functions for Repository base class
}


public class Program
{
    public static void Main()
    {

        IRepository<BaseEntity, BaseDTO> repo = new AdministratorRepository("connectionString");
        Console.WriteLine(repo.GetTestString());
        Console.WriteLine("Hello World");
    }
}
...