Решение NHibernate 3.2 LINQ n + 1 - PullRequest
4 голосов
/ 23 декабря 2011

Я использую NHibernate 3.2 с LINQ и имею n + 1 запросов на выбор в SQL.В сопоставлении fetch установлено значение «join».Когда я использую критерии, есть только один запрос выбора.Также я не могу найти метод Fetch() или FetchMany(), который я видел в разных примерах.Существует ли какой-либо способ решения проблемы n + 1 с использованием Linq для Nhibernate 3.2?
Упрощенная версия моего кода

class News
{
    public virtual int Id { get; protected set; }
    public virtual DateTime Date { get; set; }
    public virtual Category Category { get; set; }
}

class Category
{
    public virtual int Id { get; protected set; }
    public virtual ISet<News> News { get; set; }
}

Отображения:

<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2" assembly="News" namespace="NewsManagement.Models">
<class name="News" table="News">
<id name="Id">
        <generator class="native" />
</id>
<property name="Date" not-null="true" />
<many-to-one name="Category" fetch="join" column="CategoryId" class="Category, NHibernateManyToOne" not-null="true"/>
</class>
</hibernate-mapping>

<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2" assembly="News" namespace="NewsManagement.Models">
<class name="Category" table="Categories">
    <id name="Id" column="Id">
        <generator class="native" />
    </id>
    <set name="News" fetch="join" cascade="all-delete-orphan">
        <key column="CategoryId" />
        <one-to-many class="News, NHibernateOneToMany" /> 
    </set>
</class>
</hibernate-mapping>

Запрос, который вызывает много вариантов выбора:

var news = newsRepository.Linq().Skip(DefaultPageSize*currentPageIndex).Take(DefaultPageSize).OrderByDescending(x => x.Date).ToList();

Вот так нет лишних вариантов выбора:

var criteria = Session.CreateCriteria<News>();
criteria = criteria.AddOrder(Order.Desc(property));
criteria = criteria.SetMaxResults(pageSize).SetFirstResult((pageNumber - 1)*pageSize);
var news = criteria.List<News>();

Ответы [ 2 ]

6 голосов
/ 23 декабря 2011

Попробуйте с этим:

criteria.SetFetchMode("Category",FetchMode.Eager);

или с Linq:

session.Query<News>()
                .Fetch(k => k.Category)
                .Skip(n)
                .Take(m)

Или, альтернативно, с помощью QueryOver:

var result = session.QueryOver<News>
                    .Fetch(x => x.Category).Eager
                    .List();
1 голос
/ 23 декабря 2011
var result = session.Query<News>()
                    .Fetch(x => x.Category)
                    .ToList();

^ Это то, что вы ищете?

. Запрос живет в using NHibernate.Linq;.

В качестве альтернативы есть QueryOver<T>

Который будет выглядеть так:

var result = session.QueryOver<News>
                    .Fetch(x => x.Category).Eager
                    .List();
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...