nHibernate запрос «многие ко многим» с использованием API Criteria - PullRequest
5 голосов
/ 15 февраля 2010

Перед тем как спросить, я просмотрел все соответствующие сообщения на эту тему

Я также читал этот пост в блоге: http://ayende.com/Blog/archive/2007/12/23/NHiberante-Querying-Many-To-Many-associations-using-the-Criteria-API.aspx

У меня есть Команды, и у меня есть Члены, между ними есть отношения многие ко многим

В основном: Member -> MemberTeam <- Team </p>

С помощью моего запроса я пытаюсь получить всех участников, принадлежащих к одной команде, в качестве запрашиваемого участника (включая запрашиваемого участника)

Я создал свои таблицы, используя следующее FluentHibernate:

Код TeamMap:

Id(x => x.ID).GeneratedBy.GuidComb().UnsavedValue("00000000-0000-0000-0000-000000000000");

HasManyToMany(x => x.Members)
    .Table("MemberTeam")
    .ChildKeyColumn("TeamID")
    .ParentKeyColumn("MemberID");

Код членской карты:

Id(x => x.ID).GeneratedBy.GuidComb().UnsavedValue("00000000-0000-0000-0000-000000000000");

HasManyToMany(x => x.Teams)
    .Table("MemberTeam")
    .ChildKeyColumn("MemberID")
    .ParentKeyColumn("TeamID");

Код, с которым я делаю запрос:

DetachedCriteria dCriteria = DetachedCriteria.For(typeof(Team), "team")
    .SetProjection(Projections.Id())
    .Add(Property.ForName("team.ID").EqProperty("mt.ID"));

ICriteria criteria = Session.CreateCriteria(typeof (Member), "member")
    .CreateAlias("Teams", "mt")
    .Add(Subqueries.Exists(dCriteria))
    .Add(Restrictions.Eq("mt.MemberID", new Guid(memberID)));

IList<Member> list = criteria.List<Member>();

Я знаю, что я делаю что-то не так, но не могу понять, что это его

Любая помощь будет оценена

Большое спасибо!

P.S. Моя карта выглядит хорошо, я могу сохранять объекты просто отлично!

1 Ответ

4 голосов
/ 16 февраля 2010

Благодаря вызову CreateAlias для пути Teams NHibernate присоединится к соответствующим таблицам для вас. Нет необходимости выполнять подзапрос для идентификаторов участников:

var members = session
    .CreateCriteria<Member>("member")
    .CreateAlias("Teams", "mt")
    .List<Member>();

Тем не менее, вы даже не используете псевдоним, поэтому вы можете просто предпочесть получить команды:

var members = session
    .CreateCriteria<Member>("member")
    .SetFetchMode("Teams", FetchMode.Eager)
    .List<Member>();

Это гарантирует, что вы больше не попадете в базу данных при доступе к каждой коллекции Member Teams.

...