通用存储库添加自定义方法

Jes*_*hez 5 c# generics asp.net-mvc

我正在尝试创建一个带有Generic Repository的库,以便以后在我的MVC App中使用.代码如下......

public abstract class BaseEntity
{
}

public interface IEntity<T>
{
    T Id { get; set; }
}

public abstract class Entity<T> : BaseEntity, IEntity<T>
{
    public virtual T Id { get; set; }
}

public interface IAuditableEntity
{
    int? UsuarioId { get; set; }
    DateTime CreatedDate { get; set; }
    string CreatedBy { get; set; }
    DateTime UpdatedDate { get; set; }
    string UpdatedBy { get; set; }
}

public abstract class AuditableEntity<T> : Entity<T>, IAuditableEntity
{
    public int? UsuarioId { get; set; }
    public DateTime CreatedDate { get; set; }
    public string CreatedBy { get; set; }
    public DateTime UpdatedDate { get; set; }
    public string UpdatedBy { get; set; }
}   

public interface IGenericRepository<T> where T : BaseEntity
{
    IEnumerable<T> GetAll();
    IEnumerable<T> GetByUsuarioId(int usuarioId);
    T GetById(int id);
    T Add(T entity);
    T Delete(T entity);
    void Edit(T entity);
    void Save();
}

public class GenericRepository<T> : IGenericRepository<T>
   where T : BaseEntity
{
    protected DbContext _entities;
    protected readonly IDbSet<T> _dbset;

    public GenericRepository(DbContext context)
    {
        _entities = context;
        _dbset = context.Set<T>();
    }

    public virtual IEnumerable<T> GetAll()
    {
        return _dbset.AsEnumerable<T>();
    }

    public IEnumerable<T> FindBy(Expression<Func<T, bool>> predicate)
    {
        IEnumerable<T> query = _dbset.Where(predicate).AsEnumerable();
        return query;
    }


    public virtual IEnumerable<T> GetByUsuarioId(int usuarioId)
    {
        // NO SÉ COMO IMPLEMENTAR ESTE METODO!!!!!
        return null;
        // NO SÉ COMO IMPLEMENTAR ESTE METODO!!!!!
    }

    public virtual T GetById(int id)
    {
        return _dbset.Find(id);
    }

    public virtual T Add(T entity)
    {
        return _dbset.Add(entity);
    }

    public virtual T Delete(T entity)
    {
        return _dbset.Remove(entity);
    }

    public virtual void Edit(T entity)
    {
        _entities.Entry(entity).State = EntityState.Modified;
    }

    public virtual void Save()
    {
        _entities.SaveChanges();
    }
}
Run Code Online (Sandbox Code Playgroud)

这些是我的一些课程POCO ......

   public class Documento : AuditableEntity<int>
{
    public string Descripcion { get; set; }        
    public string Foto { get; set; }

    public virtual Usuario Usuario { get; set; }
}

public class Gasto : AuditableEntity<int>
{
    public int? TaxiId { get; set; }        
    public int TipoGastoId { get; set; }
    public DateTime Fecha { get; set; }
    public double Importe { get; set; }
    public int Kilometros { get; set; }
    public string Descripcion { get; set; }
    public string Foto { get; set; }

    public virtual Usuario Usuario { get; set; }
    public virtual Taxi Taxi { get; set; }
    public virtual TipoGasto TipoGasto { get; set; }
    public virtual PartidaTarjetas PartidaTarjetas { get; set; }

    public virtual ICollection<Tarea> Tareas { get; set; }      

    public int? PartidaTarjetasId { get; set; }

    public Gasto()
    {
        Tareas = new List<Tarea>();
    }
}   

public class Nivel : Entity<int>
{
    public string Descripcion { get; set; }
    public string PaginaInicio { get; set; }

    public virtual ICollection<Usuario> Usuarios { get; set; }

    public Nivel()
    {
        Usuarios = new List<Usuario>();
    }
}   
Run Code Online (Sandbox Code Playgroud)

我的问题是它不会实现该方法......

    public virtual IEnumerable<T> GetByUsuarioId(int usuarioId)
    {
        // NO SÉ COMO IMPLEMENTAR ESTE METODO!!!!!
        return null;
      // NO SÉ COMO IMPLEMENTAR ESTE METODO!!!!!
    }
Run Code Online (Sandbox Code Playgroud)

这是一个通用的方法,它应该只在IAuditable类型时返回结果...,也必须是某些东西......

    public virtual IEnumerable<T> GetByUsuarioId(int usuarioId)
    {
        return _dbset.FindBy(c => c.UsuarioId == usuarioId);
    }
Run Code Online (Sandbox Code Playgroud)

你能救我吗?谢谢.

你好C鲍尔......

注意 ...

public abstract class BaseEntity
{
}

public interface IEntity<T>
{
    T Id { get; set; }
}

public abstract class Entity<T> : BaseEntity, IEntity<T>
{
    public virtual T Id { get; set; }
}

public interface IAuditableEntity
{
    int? UsuarioId { get; set; }
    DateTime CreatedDate { get; set; }
    string CreatedBy { get; set; }
    DateTime UpdatedDate { get; set; }
    string UpdatedBy { get; set; }
}

public abstract class AuditableEntity<T> : Entity<T>, IAuditableEntity
{
    public int? UsuarioId { get; set; }
    public DateTime CreatedDate { get; set; }
    public string CreatedBy { get; set; }
    public DateTime UpdatedDate { get; set; }
    public string UpdatedBy { get; set; }
}   
Run Code Online (Sandbox Code Playgroud)

我的GenericRepositry是......

public interface IGenericRepository<T> where T : BaseEntity
Run Code Online (Sandbox Code Playgroud)

那么现在 ?

public interface IGenericRepository<T> where T : ¿¿ ??
Run Code Online (Sandbox Code Playgroud)

Dav*_*d L 0

如果这是永远正确的,请将其添加到您的基本虚拟方法中,以便在整个代码库中强制执行。

public virtual IEnumerable<T> GetByUsuarioId(int usuarioId)
{
    if (!typeof(T).GetInterfaces().Contains(typeof(IAuditable))
        return Enumerable.Empty<T>(); // per OffHeGoes' suggestion

    return _dbset.FindBy(c => c.UsuarioId == usuarioId);
}
Run Code Online (Sandbox Code Playgroud)

当然,根据下面的评论(谢谢大家),这个解决方案既存在性能问题,也存在潜在的逻辑隐藏问题。这不是解决问题的最优雅的方法,但它确实解决了问题。