科技行者

行者学院 转型私董会 科技行者专题报道 网红大战科技行者

知识库

知识库 安全导航

至顶网软件频道基础软件ASP.NET服务器控件之控件状态

ASP.NET服务器控件之控件状态

  • 扫一扫
    分享文章到微信

  • 扫一扫
    关注官方公众号
    至顶头条

我们讨论了有关视图状态的内容。部分读者已经发现,如果页面或者控件禁用了视图状态

作者:金属边缘 来源:天极开发 2007年11月6日

关键字: Windows

  • 评论
  • 分享微博
  • 分享邮件
 典型应用

  前文已经较为详细的介绍了控件状态的基本概念。本小节将通过一个示例说明控件状态的应用方法,以便加深读者对于基本概念的认识。

  示例列举了一个同时在控件状态和视图状态中保存状态的自定义控件IndexButton。在此示例中,IndexButton控件派生自Button类,还定义了一个IndexControlState属性,并将该属性值保存在控件状态中。为了进行比较,IndexButton还定义了一个IndexInViewState属性,该属性存储在ViewState字典中。控件实现具体源代码如下所示:

using System;
using System.ComponentModel;
using System.Security.Permissions;
using System.Web;
using System.Web.UI;
using System.Web.UI.WebControls;
namespace WebControlLibrary{
 [ AspNetHostingPermission(SecurityAction.Demand, Level = AspNetHostingPermissionLevel.Minimal), AspNetHostingPermission(SecurityAction.InheritanceDemand, Level = AspNetHostingPermissionLevel.Minimal), ToolboxData("<{0}:IndexButton runat=\"server\"> </{0}:IndexButton>") ]
 public class IndexButton : Button {
  // 定义私有变量
  private int _indexInControlState; //利用控件状态实现属性
  IndexInControlState [ Bindable(true), Category("Behavior"), DefaultValue(0), Description("该属性使用控件状态存储.") ]
  public int IndexInControlState {
   get { return _indexInControlState; }
   set { _indexInControlState = value; }
  } //利用视图状态实现属性
  IndexInViewState [ Bindable(true), Category("Behavior"), DefaultValue(0), Description("该属性使用视图状态存储.") ]
  public int IndexInViewState {
   get {
    object obj = ViewState["IndexInViewState"];
    return (obj == null) ? 0 : (int)obj;
   }
   set {
    ViewState["IndexInViewState"] = value;
   }
  }
  //重写OnInit方法,启用页面控件状态
  protected override void OnInit(EventArgs e) {
   base.OnInit(e);
   Page.RegisterRequiresControlState(this);
  }
  //重写SaveControlState方法,保存控件状态数据
  protected override object SaveControlState() {
   object obj = base.SaveControlState();
   if (_indexInControlState != 0) {
    if (obj != null) {
     return new Pair(obj, _indexInControlState);
    } else {
     return (_indexInControlState); }
    } else { return obj; }
   }
   //重写LoadControlState方法,加载控件状态数据
   protected override void LoadControlState(object state) {
    if (state != null) {
     Pair p = state as Pair;
     if (p != null) {
      base.LoadControlState(p.First);
      _indexInControlState = (int)p.Second;
     } else {
      if (state is int) {
       _indexInControlState = (int)state;
      } else { base.LoadControlState(state); }
     }
    }
   }
  }
 }

  如上代码实现了一个继承自Button基类的IndexButton控件,其中包括属性IndexControlState和IndexInViewState。根据代码实现可知,IndexInViewState属性利用了视图状态来存储值,而Index属性利用了控件状态来存储值。前者的实现非常简单,在此不再说明。后者的实现主要通过完成以下三个步骤,才使控件参与控件状态。

  (1)重写OnInit方法并调用RegisterRequiresControlState方法向页面注册,以参与控件状态。需要注意的是:必须针对每个请求完成此任务。

  (2)重写SaveControlState方法,以在控件状态中保存数据。

  (3)重写LoadControlState方法,以从控件状态加载数据。此方法调用基类方法,并获取基类对控件状态的基值。如果_indexInControlState字段不为零,而且基类的控件状态也不为空,Pair类便可作为方便的数据结构使用,用来保存和还原由两部分组成的控件状态。

  读者可以回想一下前一篇介绍视图状态文章中的示例。其中同样也定义了两个属性,一个是采用视图状态构建的TextInViewState属性,另一个是使用私有变量实现的Text属性。前者TextInViewState属性与上文示例中的IndexInViewState属性的实现方法几乎完全相同,其无非是利用ViewState存储属性值而已。然而,后者Text属性与上文示例的IndexInControlState属性虽然有些类似,例如,二者在实现过程中都使用了私有变量,但是,二者的本质不同。Text使用的是私有变量,而IndexInControlState使用的是控件状态,其关键是通过OnInit方法启用了控件状态功能,并重写SaveControlState和LoadControlState方法,以便自定义控件状态数据的保存和加载过程。建议读者在阅读本文的同时,也能够注意到本段所述内容。这对于理解视图状态和控件状态概念有着重要意义。

  下面列举了为测试IndexButton控件而创建的Default.aspx页面源代码。

<%@ Page Language="C#" EnableViewState="false" AutoEventWireup="true" CodeFile="Default.aspx.cs" Inherits="_Default" %>
<%@ Register Assembly="WebControlLibrary" Namespace="WebControlLibrary" TagPrefix="aspSample" %>
<script runat="server">
void Page_Load(object sender, EventArgs e) {
 Label1.Text = (IndexButton1.IndexControlState++).ToString();
 Label2.Text = (IndexButton1.IndexInViewState++).ToString();
}
</script>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head runat="server">
<title>使用视图状态和控件状态</title>
</head>
<body>
<form id="form1" runat="server">
<div> 请单击该按钮: <aspSample:IndexButton Text="IndexButton" ID="IndexButton1" runat="server" />
<br /> <br />
Index属性值是: <asp:Label ID="Label1" runat="server" Text="Label"> </asp:Label>
<br /> IndexInViewState属性值是: <asp:Label ID="Label2" runat="server" Text="Label"> </asp:Label>
<br />
</div>
</form>
</body>
</html>

  以上代码很简单。关键是读者要注意在@ Page指令中设置了EnableViewState="false",以便在页面禁用视图状态。此时,页面及页面内的所有控件,包括IndexButton都无法使用视图状态。那么,当运行该页面时应呈现怎样的应用效果呢?具体页面应用效果如图1所示。


图1

  如图1所示,当用户单击"IndexButton"按钮时,由于页面禁用了视图状态,因此,IndexInViewState属性无法完成其实际功能,其属性值将一直保持为0。然而,页面禁用视图状态对于由控件状态实现的属性IndexControlState而言,则没有丝毫影响。每当用户单击按钮一次,那么个该属性值增加1。

  通过以上示例,相信读者已经对视图状态和控件状态有了更为深入的认识。然而,可能还是有一个疑问缠绕在心中:视图状态和控件状态如此相似,那么该在何种情况下使用视图状态,又在何种情况下使用控件状态呢?通常而言,视图状态当需要存储少量回发到自身的页信息时使用。使用ViewState属性可提供具有基本安全性的功能。控件状态当需要在服务器的往返过程间存储少量控件状态信息时使用。关键的一点是:应该对那些在回发过程中,对控件至关重要的少量关键数据使用控件状态,而不要将控件状态作为视图状态的备用选项使用。

  小结

  本文主要介绍了控件状态的基本概念,并通过一个典型示例说明了这种技术特性的应用方法。需要再次提醒的是:仅对那些在回发过程中对控件至关重要的少量关键数据使用控件状态,而不要将控件状态作为视图状态的备用选项使用。这是开发人员应用视图状态和控件状态的关键所在。

查看本文来源

    • 评论
    • 分享微博
    • 分享邮件
    邮件订阅

    如果您非常迫切的想了解IT领域最新产品与技术信息,那么订阅至顶网技术邮件将是您的最佳途径之一。

    重磅专题
    往期文章
    最新文章