让GridView在大数据集时高效的分页在ASP.NET 1.1下有一个很好用的数据列表显示控件DataGrid,他有一个专门的设置数据记录数的属性"VirtualItemCount",这样可以很好的解决大数据集的分页问题,即每次读取一页的数据,但还是把数据集总数值赋给这个虚拟记录数属性,也就避免了每次都要读取所有数据 (当然你也可以自定义分页方式,也有很多现成的分页控件,这就不是本文讨论的范围了, 我比较懒,觉得GridView自身的分页功能已经足够用了,呵呵). 到了ASP.NET 2.0,有了更强大的数据显示控件GridView, 但有一个遗憾,它没有了"VirtualItemCount"这个属性, PageCount属性也是只读的,如果你想用它本身的分页功能,只能把所有数据作为它的数据源, 当这个数据集很大时,性能或许会成为一个问题, 或者这样让追求效率的你觉得有些不爽,怎么办?
有一个变通的方法可以解决这个问题:ObjectDataSource.
1. 先需要一个配合ObjectDataSource使用的数据分页设配器:
/**/
/// <summary>
/// 分页适配器。
/// </summary>
internal class GridViewPagingAdapter
... {
private object m_dataList;
private int m_virtualItemCount;
// 构造一个适配器。
public GridViewPagingAdapter(object list, int rowCount)
...{
m_dataList = list;
m_virtualItemCount = rowCount;
}
// 取全部的数据(其实也只是一页的数据啦)。
public object GetData()
...{
return m_dataList;
}
// 返回你设定的记录数。
public int VirtualItemCount()
...{
return m_virtualItemCount;
}
// 取一页的数据。
public object GetData(int startRow, int maxRows)
...{
return m_dataList;
}
}
/// 分页适配器。
/// </summary>
internal class GridViewPagingAdapter
... {
private object m_dataList;
private int m_virtualItemCount;
// 构造一个适配器。
public GridViewPagingAdapter(object list, int rowCount)
...{
m_dataList = list;
m_virtualItemCount = rowCount;
}
// 取全部的数据(其实也只是一页的数据啦)。
public object GetData()
...{
return m_dataList;
}
// 返回你设定的记录数。
public int VirtualItemCount()
...{
return m_virtualItemCount;
}
// 取一页的数据。
public object GetData(int startRow, int maxRows)
...{
return m_dataList;
}
}
2. 然后借用一下ObjectDataSource的躯体.
/**/
/// <summary>
/// 分页增强的ObjectDataSource.
/// </summary>
public class VirtualObjectDataSource : ObjectDataSource
... {
private GridViewPagingAdapter m_pagingAdapter;
public VirtualObjectDataSource(object dataSource, int virtualItemCount)
...{
// 设置数据实体类型,可不设(它自己会辨认的)
this.DataObjectTypeName = string.Empty;
// 当然要分页的,要不然就没必要这么做了.
this.EnablePaging = true;
// 页数是虚拟的,自然ViewState就没有意义了。
this.EnableViewState = false;
/**//// 设置分页需要的属性和方法.
this.TypeName = "SSW.Framework.Web.UI.GridViewPagingAdapter";
this.SelectMethod = "GetData";
this.SelectCountMethod = "VirtualItemCount";
this.StartRowIndexParameterName = "startRow";
this.MaximumRowsParameterName = "maxRows";
// 生成一个分页设配器.
this.m_pagingAdapter = new GridViewPagingAdapter(dataSource, virtualItemCount);
// 关键在这里 - 把已构造好的业分页设配器替换掉自身的数据源。
this.ObjectCreating += new ObjectDataSourceObjectEventHandler(ObjectDataSourceExtension_ObjectCreating);
}
void ObjectDataSourceExtension_ObjectCreating(object sender, ObjectDataSourceEventArgs e)
...{
e.ObjectInstance = this.m_pagingAdapter;
}
}
/// 分页增强的ObjectDataSource.
/// </summary>
public class VirtualObjectDataSource : ObjectDataSource
... {
private GridViewPagingAdapter m_pagingAdapter;
public VirtualObjectDataSource(object dataSource, int virtualItemCount)
...{
// 设置数据实体类型,可不设(它自己会辨认的)
this.DataObjectTypeName = string.Empty;
// 当然要分页的,要不然就没必要这么做了.
this.EnablePaging = true;
// 页数是虚拟的,自然ViewState就没有意义了。
this.EnableViewState = false;
/**//// 设置分页需要的属性和方法.
this.TypeName = "SSW.Framework.Web.UI.GridViewPagingAdapter";
this.SelectMethod = "GetData";
this.SelectCountMethod = "VirtualItemCount";
this.StartRowIndexParameterName = "startRow";
this.MaximumRowsParameterName = "maxRows";
// 生成一个分页设配器.
this.m_pagingAdapter = new GridViewPagingAdapter(dataSource, virtualItemCount);
// 关键在这里 - 把已构造好的业分页设配器替换掉自身的数据源。
this.ObjectCreating += new ObjectDataSourceObjectEventHandler(ObjectDataSourceExtension_ObjectCreating);
}
void ObjectDataSourceExtension_ObjectCreating(object sender, ObjectDataSourceEventArgs e)
...{
e.ObjectInstance = this.m_pagingAdapter;
}
}
3.然后就简单了- 把VirtualObjectDataSource绑定到GridView上就完成了.
Protected
Sub FillGridView()
Sub FillGridView()
//得到数据和构造一个VirtualObjectDataSource.
Dim list As IList(Of T) = GetDataSource(Me.SearchResultGrid.PageIndex)
Dim rowCount As Integer = GetRowCount()
Dim ods As VirtualObjectDataSource = New (list, rowCount)
// 数据绑定.
Me.SearchResultGrid.DataSource = ods
Me.SearchResultGrid.DataBind()
//完成!
End Sub
//得到数据和构造一个VirtualObjectDataSource.
Dim list As IList(Of T) = GetDataSource(Me.SearchResultGrid.PageIndex)
Dim rowCount As Integer = GetRowCount()
Dim ods As VirtualObjectDataSource = New (list, rowCount)
// 数据绑定.
Me.SearchResultGrid.DataSource = ods
Me.SearchResultGrid.DataBind()
//完成!
End Sub
对于Huge的数据集绑定显示, 可以让Web服务器和数据库服务器减轻一些压力了. :)
附:终于完成了第一篇文章,心情十分轻松,争取以后多写一些Tip与你分享. ^_~