去掉DataGrid的ViewState中的无用数据

2024-02-09 17:58

本文主要是介绍去掉DataGrid的ViewState中的无用数据,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、             DataGrid控件中的ViewState

   Asp.Net中,我们使用最多的恐怕就是DataGrid列表控件了。这个控件的功能的却非常强大,也非常好用。它不但可以实现任意的格式化选择,而且还可以动态进行分页、排序、添加按钮、动态编辑等功能。可以说,DataGrid控件已经实现了我们需要的大部分功能。

   遗憾的是,上面说的很多非常实用的功能,大部分需要ViewState的支持,就是说DataGrid控件需要通过ViewState来保存控件的状态的,如果我们关闭了ViewState,即在HTML代码中使用了EnableViewState="False"属性,那么,上面的所有有用的功能我们将无法使用。

   ViewState的缺点也是很大的,就是DataGrid会把所有的数据源中的数据存放到ViewState中。DataGrid是一个列表控件,它存放在ViewState中的数据包括了列表中所有单元格(Cell)中的数据,更加可恨的是,即时我们打开了分页功能,它也要把所有的没有显示出的数据源中的数据放到ViewState里面,如果不在数据库端控制数据量大小的话,DataGrid控件实际上是把数据库表中所有数据统统都存放在ViewState中。直接后果就是Asp.Net生成发送到客户端的源Html代码异常庞大。

   大家知道,Asp.Net中,ViewState实际上是一个名称/值对的字典,为了正确在各种编码的网页中保存,将ViewState序列化后又进行了Base64编码。这将更加加大ViewState的数据量。

   我试验了一下,一个包括5000条记录的数据库,如果不在数据库端控制数据量的话,直接使用Select * From TableName这样的Sql语句绑定数据的话,一个Asp.Net动态网页(仅有一个DataGrid控件的测试页)的ViewState60k之多。而这样一个网页放到公共Web网站,缓慢的网页下载速度将会使大部分人望而却步,使用56k猫上网的人就更不要提了,根本无法访问这样的网页,等待的时间将不可忍受。

   所以大部分有Asp.Net开发经验的开发者,如果不使用DataGrid的高级功能的话,一般是把EnableViewState="False"属性给加入的,还有就是使用自己的分页控件,不使用DataGrid提供的分页功能,这将取消DataGridViewState,大大减少最终生成的Html的代码量。

 

二、DataGrid中必须使用ViewState的一些功能

   不错,很多情况下我们仅需要DataGrid的显示列表数据的功能,或者使用自己的分页功能,这种情况下可以使用EnableViewState="False"属性,但是如果我们需要使用DataGrid的动态编辑、按钮列,或者要访问DataGrid的索引项时,我们必须把DataGridViewState打开,这将不可避免的出现ViewState数据量过大的现象,但是我们没有办法。

   我大略统计了一下DataGrid中需要使用ViewState的一些功能属性,它们都必须使用ViewState,而这些属性我们很多情况下都必须使用:

DataGrid.CurrentPageIndex                     DataGrid分页的页索引

DataGrid.DataKeys                             取每个记录的键值

DataGrid.EditItemIndex                        编辑项的索引

DataGrid.Items                                DataGrid中行的集合

DataGrid.PageCount                            DataGrid分页的总页数

 

而在使用DataGrid内置事件时,传递事件参数的DataGridItemEventArgsDataGridCommandEventArgsDataGridSortCommandEventArgsDataGridPageChangedEventArgs类更加要使用DataGridViewState,否则它们将无法工作。

   这里我给一个简单的示例,在DataGrid中加入一个CheckBox列,这使当页面回发的时候我可以获取DataGrid的键值。但是如果关闭ViewState的话,此功能无法使用。

 

 

Html

<Asp:DataGrid ID=”myDataGrid” Runat=”Server” DataKeyField=”ID” AutoGenerateColumns=”False”>

    <Columns>

       <asp:TemplateColumn>

           <ItemTemplate>

              <Asp:CheckBox ID=”myBox” Runat=”Server” />

           </ ItemTemplate>

       </ asp:TemplateColumn>

    </Columns>

</Asp:DataGrid>

<br /><br />

<Asp:Button ID=”bntSubmit” Runat=”Server” Text=”OK” />

 

Code

private void btnSubmit_Click(object sender, System.EventArgs e)

{

    CheckBox cb;

    string str;

    for(int i=0;i<myDataGrid.tems.Count;i++)

    {

       cb = (CheckBox)myDataGrid.Items[i].Cells[2].FindControl("myBox");

       if(cb.Checked)

           str += myDataGrid.DataKeys[myDataGrid.Items[i].ItemIndex].ToString();

    }

    // 此时str变量就是DataGrid中选中行的数据库键值。

}

 

上面的代码使用很成功,不过如果在DataGrid中使用了EnableViewState=”False”属性,那么上面的代码根本无法起作用。

 

三、去掉ViewStateDataGrid无用数据

   终于说到正题了,我这篇文章说的就是怎样去掉ViewState中的无用数据,而保留有用的数据。文章写到这里,我们可以很明显的看出来,DataGrid保存在ViewState中的数据分为两个部分,一部分是保存索引用的,就是DataKeysDataItems这样的属性使用的数据,我们把它称之为索引数据。还有一部分是DataGrid中数据源的内容,我们称之为列表数据。

  我们如果把实际上无用的列表数据从ViewState中去除,这样可以大大减小页面ViewState的数据大小,使用DataGridViewState数据量太大的根本原因就是列表数据存放在ViewState中。

  我在微软的.Net Framework SDK文档中没有找到关闭列表数据ViewState的任何内容,就是说微软没有给出DataGrid运行时的任何调用顺序和内部的工作机制。没办法,我使用了一个工具来获得DataGrid的内部工作流程,发现它在数据绑定初始化的时候,生成了一个叫DataGridTable的控件对象,这个对象是继承System.Web.UI.WebControls.Table控件的。而且这个对象是最先加入(使用Controls.Add()方法)DataGrid中的。而且ViewState中的DataGrid列表数据也是这个控件加入到DataGrid中的。实际上,ViewState中的DataGrid的单元格中的数据实际上是System.Web.UI.WebControls.Table控件的SaveViewState()方法给加进去的。这些数据很多情况下是不需要的。

   好,原因找到了,解决问题就好办了,因为微软已经给出了控制DataGrid中子控件的方法。我们既然知道DataGridTable控件是DataGrid中最先生成的控件,那么我们通过DataGrid.Controls属性就可以直接获取DataGrid中子控件的引用,获得引用后就可以控制子控件了。解决方法就是在数据绑定的时候,设置DataGridDataGridTable控件的EnableViewState属性为False就可以了。

代码如下:

首先必须重写DataGrid.ItemDataBoundDataGrid.ItemCreated事件,我们用它们来控制在向客户端写html之前DataGrid的动作。这两个事件任选其一,都可以实现效果。我们使用ItemDataBound事件来写例子,DataGrid示例IDmyDataGrid

[C#]

首先在页面初始化中的InitializeComponent()方法内加入事件的委托:

private void InitializeComponent()

{

    this.myDataGrid.ItemDataBound +=

new DataGridItemEventHandler(this.myDataGrid_ItemDataBound);

}

然后在myDataGrid_ItemDataBound方法内加入控制代码:

private void myDataGrid_ItemDataBound(object sender, DataGridItemEventArgs e)

{

    myDataGrid.Controls[0].EnableViewState = false;

}

 

[VB.Net]

Private Sub myDataGrid_ItemDataBound(sender As Object,e As DataGridItemEventArgs) _

Handles myDataGrid.ItemDataBound

 

    myDataGrid.Contols.Item(0).EnableViewState = False

 

End Sub

 

好了,使用DataGrid时,把上面的代码加入,将减小使用DataGridViewState90%的数据量。而且,DataGrid中许多使用ViewState的功能丝毫不少,岂不是两全其美?

 

注意:

1.上面说的法子有一种情况下不能使用,就是使用DataGrid的内部分页功能时,重写DataGrid.PageIndexChanged事件时,调用DataGridPageChangedEventArgs时,必须把所有的ViewState打开,包括列表数据,关闭任何的ViewState,都将导致DataGridPageChangedEventArgs的索引丢失,无法分页。

不过这个缺点很好解决,很多人已经写了自定义的分页控件,这些控件是不需要DataGrid的任何数据的。可以提供非常完美的分页功能。

2. (重要)我春节的这几天对上面说的办法进行了详细的测试,发现上面的办法确实有效,可以大大的减少aspx页面的ViewState的数据量,但是这个办法在使用上有一个注意的地方,否则会出现数据无法显示的毛病。

我一直在奇怪,微软写DataGrid为什么要加入一个DataGridTable类,而且还要把所有的数据库数据存放在ViewState中。经过测试我才发现,这是因为有了System.Web.UI.Page.IsPostBack属性的缘故。我们经常喜欢利用Page.IsPostBack属性检测页面是否是第一次运行,如下代码:

private void Page_Load(object sender, System.EventArgs e)

{

    if(!Page.IsPostBack)

    {

       // 页面第一次运行,执行数据绑定

       SqlConnection conn = new SqlConnection(“...”);

       SqlCommand comm = new SqlCommand(“...”,conn);

       ...

       ...

       myDataGrid.DataBind() // 数据绑定

    }

}

当然,网页执行第一次时,DataGrid的内容正常显示,而使用了上面的去除ViewState方法后,页面如果回发处理,DataGrid的内容将会消失。我才明白DataGridDataGridTable把数据存放在ViewState内的用意。微软的设计是非常严谨的,他们的用意就是当使用Page.IsPostBack属性时,仅访问一次数据库就可以永久保持DataGrid的数据(在不离开此页面的情况下),数据存放的地点就是页面的ViewState中。这样页面回发后,DataGrid就可以从ViewState中重新生成DataGrid的显示内容,无需访问数据库。所以说微软以牺牲客户下载的速度(ViewState数据量)来保证服务器的资源,大家都知道频繁访问数据库对服务器的资源消耗很大。

所以,使用上面减少DataGridViewState数据的法子是可行的,但是必须使所有的页面回发处理都必须进行数据绑定,否则DataGrid无法获得数据库内容,也无法获得ViewState中保存的数据,那么回发后DataGrid将无法显示任何内容。

总结,使用上面减少ViewState的办法可以大大加快客户端的下载显示速度,但是频繁的数据库访问将加大服务器的压力;使用ViewState可以减轻服务器的压力,但是又加大了客户端的下载时间,它们是互相矛盾的。所以开发者要根据实际情况选择是否使用DataGrid.Controls[0].EnableViewState=false;的法子,如何选择,大家请自己斟酌。

 

 

作者Blog: http://blog.csdn.net/TomMax/

这篇关于去掉DataGrid的ViewState中的无用数据的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/694912

相关文章

Python Pandas高效处理Excel数据完整指南

《PythonPandas高效处理Excel数据完整指南》在数据驱动的时代,Excel仍是大量企业存储核心数据的工具,Python的Pandas库凭借其向量化计算、内存优化和丰富的数据处理接口,成为... 目录一、环境搭建与数据读取1.1 基础环境配置1.2 数据高效载入技巧二、数据清洗核心战术2.1 缺失

Python处理超大规模数据的4大方法详解

《Python处理超大规模数据的4大方法详解》在数据的奇妙世界里,数据量就像滚雪球一样,越变越大,从最初的GB级别的小数据堆,逐渐演变成TB级别的数据大山,所以本文我们就来看看Python处理... 目录1. Mars:数据处理界的 “变形金刚”2. Dask:分布式计算的 “指挥家”3. CuPy:GPU

使用Vue-ECharts实现数据可视化图表功能

《使用Vue-ECharts实现数据可视化图表功能》在前端开发中,经常会遇到需要展示数据可视化的需求,比如柱状图、折线图、饼图等,这类需求不仅要求我们准确地将数据呈现出来,还需要兼顾美观与交互体验,所... 目录前言为什么选择 vue-ECharts?1. 基于 ECharts,功能强大2. 更符合 Vue

Java如何根据word模板导出数据

《Java如何根据word模板导出数据》这篇文章主要为大家详细介绍了Java如何实现根据word模板导出数据,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... pom.XML文件导入依赖 <dependency> <groupId>cn.afterturn</groupId>

Python实现获取带合并单元格的表格数据

《Python实现获取带合并单元格的表格数据》由于在日常运维中经常出现一些合并单元格的表格,如果要获取数据比较麻烦,所以本文我们就来聊聊如何使用Python实现获取带合并单元格的表格数据吧... 由于在日常运维中经常出现一些合并单元格的表格,如果要获取数据比较麻烦,现将将封装成类,并通过调用list_exc

Mysql数据库中数据的操作CRUD详解

《Mysql数据库中数据的操作CRUD详解》:本文主要介绍Mysql数据库中数据的操作(CRUD),详细描述对Mysql数据库中数据的操作(CRUD),包括插入、修改、删除数据,还有查询数据,包括... 目录一、插入数据(insert)1.插入数据的语法2.注意事项二、修改数据(update)1.语法2.有

SpringBoot实现接口数据加解密的三种实战方案

《SpringBoot实现接口数据加解密的三种实战方案》在金融支付、用户隐私信息传输等场景中,接口数据若以明文传输,极易被中间人攻击窃取,SpringBoot提供了多种优雅的加解密实现方案,本文将从原... 目录一、为什么需要接口数据加解密?二、核心加解密算法选择1. 对称加密(AES)2. 非对称加密(R

详解如何在SpringBoot控制器中处理用户数据

《详解如何在SpringBoot控制器中处理用户数据》在SpringBoot应用开发中,控制器(Controller)扮演着至关重要的角色,它负责接收用户请求、处理数据并返回响应,本文将深入浅出地讲解... 目录一、获取请求参数1.1 获取查询参数1.2 获取路径参数二、处理表单提交2.1 处理表单数据三、

Spring Validation中9个数据校验工具使用指南

《SpringValidation中9个数据校验工具使用指南》SpringValidation作为Spring生态系统的重要组成部分,提供了一套强大而灵活的数据校验机制,本文给大家介绍了Spring... 目录1. Bean Validation基础注解常用注解示例在控制器中应用2. 自定义约束验证器定义自

C#实现高性能Excel百万数据导出优化实战指南

《C#实现高性能Excel百万数据导出优化实战指南》在日常工作中,Excel数据导出是一个常见的需求,然而,当数据量较大时,性能和内存问题往往会成为限制导出效率的瓶颈,下面我们看看C#如何结合EPPl... 目录一、技术方案核心对比二、各方案选型建议三、性能对比数据四、核心代码实现1. MiniExcel