在禁用视图状态的情况下仍然使用ViewState对象

    技术2022-05-11  12

    在禁用视图状态的情况下仍然使用ViewState对象

    本文节选自《庖丁解牛:纵向切入ASP.NET 3.5控件和组件开发技术》一书

            当开发人员禁用了页面或控件视图状态时。控件开发人员一般在无奈情况下会使用控件状态,要重写LoadControlState,SaveControlState,还有一个OnInit方法,这样固然完全可以实现控件重要数据的控件状态数据保存。但一般LoadControlState和SaveControlState方法都要开发人员自定义编程,比较麻烦,更适合对控件中复杂的自定义类型数据进行对象序列化操作,比如一些简单的类型如:string,int,bool,color,datetime,byte,arraylist等如果是控件重要属性的话,也都挤到LoadControlState 和 SaveControlState方法中,则会显得比较啰唆。想一下,如果页面视图没有禁用时该多好,可以以this.ViewState["Text"]的格式直接使用Control中的ViewState对象,非常方便。既然ViewState对象不管禁用还是不禁用都在Control类中是存在的,那不用它岂不浪费。这一节主要内容就是如何实现在禁用视图状态下仍然可以使用ViewState对象。该功能在主控件ControlStateControl中已经实现了。回顾一下主控件的代码片段:/// <summary>/// 获得本书更多内容,请看:/// http://blog.csdn.net/ChengKing/archive/2008/08/18/2792440.aspx/// </summary>public class ControlStateControl : WebControl{    //… …    [Description("使用ViewState属性来存储数据此属性")]    public string Text_ViewState    {        get        {            String s = (String)ViewState["Text_ViewState"];            return ((s == null) ? String.Empty : s);

            }

            set        {            ViewState["Text_ViewState"] = value;        }    }    //… …    protected override object SaveControlState()    {        Pair p = new Pair();        p.First = base.SaveViewState();        p.Second = ((IStateManager)FaceStyle).SaveViewState();        //… …        return p;    }

        protected override void LoadControlState(object savedState)    {        if (savedState == null)        {            base.LoadViewState(null);            return;        }        else        {            //… …            base.LoadViewState(p.First);            //… …        }    }    //… …}在ControlStateControl中,属性Text_ViewState仍然存储在基类Control的ViewState对象中,且运行页面中同时禁用了页面和控件视图状态,但该属性仍然能够正确地应用视图状态。在视图状态启用状态下,在LoadViewState方法中会默认调用基类的base.LoadViewState方法,其中就包含对基类中ViewState对象进行对象序列化的代码,如下:protected virtual void LoadViewState(object savedState){    if (savedState != null)    {        this.ViewState.LoadViewState(savedState);        object obj2 = this.ViewState["Visible"];        if (obj2 != null)        {            if (!((bool)obj2))            {                this.flags.Set(0x10);            }            else            {                this.flags.Clear(0x10);            }            this.flags.Set(0x20);        }    }}其中这句this.ViewState.LoadViewState(savedState)为关键语句,还记得ViewState属性实际的类型为StateBag类,它是系统定义的类型视图状态实现类(我们在6.2.3小节探讨过)。同样在,SaveViewState方法中也有序列化保存ViewState属性对象的代码,如下所示:protected virtual object SaveViewState(){    if (this.flags[0x20])    {        this.ViewState["Visible"] = !this.flags[0x10];    }    if (this._viewState != null)    {        return this._viewState.SaveViewState();    }    return null;}关键语句为this._viewState.SaveViewState(),ViewState是对外属性,其操作的变量就是StateBag类型的_viewState。在视图状态被禁用的情况下,由于LoadViewState和SaveViewState不再被页框架调用(默认情况下是base.SaveViewState和base.LoadViewState方法不会再被调用),所以ViewState属性功能也就失效。了解了ViewState对象的来龙去脉,现在就讲解一下在ControlStateControl控件中仍然可以使用ViewState的原因。在LoadControlState和SaveControlState方法中分别调用base.LoadViewState和base.SaveControlState,我们可以手动调用ViewState属性对象的对象正反序列化过程。归根到底,也就是说开发人员所谓的禁用视图实际上是禁止LoadViewState和SaveViewState两个方法的执行,但理论上我们只要启动控件状态,并把这两个方法的逻辑放到LoadControlState和SaveControlState中,仍然可以利用ViewState。本节内容有些乖张,违反了ASP.NET设计页面状态的规则。既然ASP.NET框架把视图和控件状态已经分开了,建议在实际开发中分开处理,不要滥用。在LoadControlState方法中尽量只写控件状态相关逻辑,在LoadViewState中只写视图状态相关逻辑,毕竟ControlState是专门为存储控件必需的少量数据设计的。不过上面在ControlState中使用ViewState的确是非常方便的,在处理基本类型的属性时能够节省开发时间。


    最新回复(0)