Как активно загружать дочерние объекты, используя шаблон репозитория - PullRequest
6 голосов
/ 17 марта 2011

У меня есть сущность с именем Tour, которая может иметь много Agents. Я могу добавить агентов, но не могу их удалить.

// _repo is injected....
var tour = _repo.GetById(tourId);
tour.AddAgent(new Agent(tour.TourId));

Когда я пытаюсь вызвать метод Tour.RemoveAgent(), на самом деле ничего не удаляется. Я устанавливаю точку останова внутри метода Tour.RemoveAgent() и вижу, что свойство _agents имеет счетчик 0.

tour.RemoveAgent(agentId); // This doesn't work because _agents is empty

Должен ли я сделать что-то особенное для EF, чтобы заполнить свойство _agents, когда я получаю Tour из своего хранилища?

ОБНОВЛЕНИЕ: ПРОБЛЕМА РЕШЕНА (благодаря ответу Павла)

Я решил просто создать репозиторий, уникальный для каждого агрегата, чтобы было легко определить, что именно нужно включить, используя функцию Include(). Это пример, где я наследую класс GenericRepository<T> (который также включен в конец этого вопроса).

public class TourRepository : GenericRepository<Tour>
{
    public TourRepository(IDatabaseFactory databaseFactory) : base (databaseFactory)
    {
    }

    public override Tour GetById(Guid id)
    {
        return dataContext.Tours
                .Include(x => x.Agents)
                .Single(x => x.TourId == id);
    }
}

Тур класс

public partial class Tour
{
  public Guid TourId { get; private set; }
  protected virtual List<Agent> _agents { get; set; }

  public Tour()
  {
    TourId = Guid.NewGuid();
    _agents = new List<Agent>();
  }

  public void AddAgent(Agent agent)
  {
    _agents.Add(agent);
  }

  public void RemoveAgent(Guid agentId)
  {
    _agents.RemoveAll(a => a.AgentId == agentId);
  }
}

Класс агента

public partial class Agent
{
  public Guid AgentId { get; private set; }
  public Guid TourId { get; private set; }
  public Tour Tour { get; private set; }

  public Agent(Guid tourId)
  {
    TourId = tourId;
    AgentId = Guid.NewGuid();
  }
}

OnModelCreating

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
  // AGENTS ============================
  modelBuilder.Entity<Agent>()
              .HasKey(x => x.AgentId)
              .Property(p => p.AgentId);

  modelBuilder.Entity<Agent>()
              .HasRequired(p => p.Tour)
              .WithMany(t => t.Agents);

  // TOURS =============================
  modelBuilder.Entity<Tour>()
              .HasKey(x => x.TourId)
              .Property(x => x.TourId);
}

Класс репозитория

public class GenericRepository<T> : IRepository<T> where T : class {
  private MyContext dataContext;
  private readonly IDbSet<T> dbset;  

  public GenericRepository(IDatabaseFactory databaseFactory)
  {
    DatabaseFactory = databaseFactory;
    dbset = DataContext.Set<T>();
  }

  protected IDatabaseFactory DatabaseFactory
  {
    get;
    private set;
  }

  protected MyContext DataContext
  {
    get { return dataContext ?? (dataContext = DatabaseFactory.Get()); }
  }

  // ... stuff removed for brevity ...

  public T GetById(Guid id)
  {
    return dbset.Find(id);
  }
}

1 Ответ

7 голосов
/ 17 марта 2011

Попробуйте создать защищенный виртуальный список _agents {get; задавать; } public

public virtual List<Agent> _agents { get; set; }

Вы также можете загрузить, сделав что-то вроде этого:

_databaseContext.Tours.Include(x => x.Agents).Single(x => x.TourId == tourId) 

Вы можете прочитать больше здесь: http://blogs.msdn.com/b/adonet/archive/2011/01/31/using-dbcontext-in-ef-feature-ctp5-part-6-loading-related-entities.aspx

...