让GridView在大数据集时高效的分页
让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; } }
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; } }
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
对于Huge的数据集绑定显示, 可以让Web服务器和数据库服务器减轻一些压力了. :) 附:终于完成了第一篇文章,心情十分轻松,争取以后多写一些Tip与你分享. ^_~