回答:
ICriteria
には、SetFirstResult(int i)
取得する最初のアイテムのインデックスを示すメソッドがあります(基本的にはページの最初のデータ行)。
またSetMaxResults(int i)
、取得する行の数(つまり、ページサイズ)を示すメソッドもあります。
たとえば、次の基準オブジェクトは、データグリッドの最初の10件の結果を取得します。
criteria.SetFirstResult(0).SetMaxResults(10);
また、NHibernateのFutures機能を利用してクエリを実行し、合計レコード数と単一のクエリの実際の結果を取得することもできます。
例
// Get the total row count in the database.
var rowCount = this.Session.CreateCriteria(typeof(EventLogEntry))
.Add(Expression.Between("Timestamp", startDate, endDate))
.SetProjection(Projections.RowCount()).FutureValue<Int32>();
// Get the actual log entries, respecting the paging.
var results = this.Session.CreateCriteria(typeof(EventLogEntry))
.Add(Expression.Between("Timestamp", startDate, endDate))
.SetFirstResult(pageIndex * pageSize)
.SetMaxResults(pageSize)
.Future<EventLogEntry>();
合計レコード数を取得するには、次のようにします。
int iRowCount = rowCount.Value;
NHibernate 3以降では、以下を使用できますQueryOver<T>
。
var pageRecords = nhSession.QueryOver<TEntity>()
.Skip((PageNumber - 1) * PageSize)
.Take(PageSize)
.List();
次のように結果を明示的に並べることもできます。
var pageRecords = nhSession.QueryOver<TEntity>()
.OrderBy(t => t.AnOrderFieldLikeDate).Desc
.Skip((PageNumber - 1) * PageSize)
.Take(PageSize)
.List();
.Skip(PageNumber * PageSize)
この方法では、ページサイズが10の場合、最初の10行は取得されません。数式を正しくするために編集しています。概念的に、と仮定するとPageNumber
0にすべきではないことはそれが1、最低でなければなりません
public IList<Customer> GetPagedData(int page, int pageSize, out long count)
{
try
{
var all = new List<Customer>();
ISession s = NHibernateHttpModule.CurrentSession;
IList results = s.CreateMultiCriteria()
.Add(s.CreateCriteria(typeof(Customer)).SetFirstResult(page * pageSize).SetMaxResults(pageSize))
.Add(s.CreateCriteria(typeof(Customer)).SetProjection(Projections.RowCountInt64()))
.List();
foreach (var o in (IList)results[0])
all.Add((Customer)o);
count = (long)((IList)results[1])[0];
return all;
}
catch (Exception ex) { throw new Exception("GetPagedData Customer da hata", ex); }
}
データのページングがMultiCriteriaから型付けされた結果を取得する別の方法がある場合、または誰もが私のように同じことをしますか?
ありがとう
Ayendeによるこのブログ投稿で説明されているように、Linq to NHibernateを使用するのはどうですか?
コードサンプル:
(from c in nwnd.Customers select c.CustomerID)
.Skip(10).Take(10).ToList();
また、ページングの実装を含む、NHibernateを使用したデータアクセスに関するNHibernateチームのブログによる詳細な投稿を次に示します。
ほとんどの場合、GridViewでは、データのスライスと、クエリに一致したデータの総量の行の総数(rowcount)を表示する必要があります。
選択カウント(*)クエリと.SetFirstResult(n).SetMaxResult(m)クエリの両方を1回の呼び出しでデータベースに送信するには、マルチクエリを使用する必要があります。
結果は、データスライス用とカウント用の2つのリストを含むリストになります。
例:
IMultiQuery multiQuery = s.CreateMultiQuery()
.Add(s.CreateQuery("from Item i where i.Id > ?")
.SetInt32(0, 50).SetFirstResult(10))
.Add(s.CreateQuery("select count(*) from Item i where i.Id > ?")
.SetInt32(0, 50));
IList results = multiQuery.List();
IList items = (IList)results[0];
long count = (long)((IList)results[1])[0];
ページネーションに対処するための特定の構造を作成することをお勧めします。次のようなものです(私はJavaプログラマですが、簡単にマッピングできるはずです)。
public class Page {
private List results;
private int pageSize;
private int page;
public Page(Query query, int page, int pageSize) {
this.page = page;
this.pageSize = pageSize;
results = query.setFirstResult(page * pageSize)
.setMaxResults(pageSize+1)
.list();
}
public List getNextPage()
public List getPreviousPage()
public int getPageCount()
public int getCurrentPage()
public void setPageSize()
}
私は実装を提供しませんでしたが、@Jonによって提案されたメソッドを使用できます。ここでは、あなたが見てみるのに良い議論があります。