NHibernate でページングを行うにはどうすればよいですか?
-
09-06-2019 - |
質問
たとえば、ASP.NET Web ページのグリッドビュー コントロールに、表示される行数に必要なデータのみを入力したいとします。NHibernate はこれをどのようにサポートできるのでしょうか?
解決
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;
先物があなたに与えるものについての良い議論は次のとおりです。 ここ.
Hibernate 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();
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 から型指定された結果を取得する別の方法はありますか、それとも誰もが私と同じようにしますか?
ありがとう
で説明したように、Linq to NHibernate を使用してみてはいかがでしょうか。 このブログ投稿 アイエンデで?
コードサンプル:
(from c in nwnd.Customers select c.CustomerID)
.Skip(10).Take(10).ToList();
NHibernate チームのブログによる詳細な投稿は次のとおりです。 Hibernate を使用したデータ アクセス ページングの実装も含まれます。
GridView では、データのスライスと、クエリに一致したデータの総量の合計行数 (rowcount) を表示する必要があると考えられます。
MultiQuery を使用して、Select count(*) クエリと .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()
}
私は実装を提供しませんでしたが、提案されているメソッドを使用できます。 @ジョン. 。ここにあります 良い議論 見てください。