C#强化系列文章一:ViewState使用兼谈序列化

简介:
ViewState的使用比较简单,一两句话就可以了。
赋值: ViewState[key] = value;
取值:value = ViewState[key];

最主要的作用就是可以在当前页面保存值,ASP.NET的页面状态维护就是使用ViewState来实现的,基本上每一个ASPX页面都可以看到如下类似的html代码:
< input  type ="hidden"  name ="__VIEWSTATE"  id ="__VIEWSTATE"  value ="/wEPDwUKMTkwNjc4NTIwMWRkyv4ncofW5vaWXdXRtXfXn3RYQR4="   />
也就是说ViewState中的值实际上都是通过一个hidden来保存的,hidden的name为 __VIEWSTATE ,那么如果页面上有另外一个控件的名称也叫: __VIEWSTATE的话,会导致页面出错。
其实在我们进行页面开发或者进行自定义控件开发的时候,都可以使用ViewState,很方便。

ViewState是ASP.NET中特有的,相对于Session来说,它保存的值只能在当前页面使用,并且保存的只能是已经序列化的类,比如.NET中的strings, integers, Booleans, arrays, ArrayList, hashtable,DataTable等。
那么如何将自定义的类放入ViewState中呢,这个就涉及到如下所说的序列化的问题了:
序列化简单来说就是把一个对象转化成一种可以持久保存的数据,当下次需要使用时再把之前保存的数据反序列化成一个对象。
当然在.NET中提供了简便的方法进行序列化的操作。
下面我以一个简单的例子来说明
将自定义类Test保存到viewstate中的按钮事件代码:
     protected   void  Button1_Click( object  sender, EventArgs e)
    
{
        Test test 
= ViewState["VIEW_TEST"as Test;
        
if (test == null)
        
{
            test 
= new Test();
        }

        test[
-1= TextBox1.Text;

        ViewState[
"VIEW_TEST"= test;
    }

下面再看一下自定义类Test的实现:
[SerializableAttribute]
class  Test
{
    
private IList list;

    
public Test()
    
{
        list 
= new ArrayList();
    }


    
public object this[int index]
    
{
        
get
        
{
            
if (index >= list.Count)
            
{
                
return null;
            }

            
return list[index];
        }

        
set
        
{
            list.Add(value);
        }


    }

}
特别注意第一行的 SerializableAttribute属性,指定这个属性后就代表此类是可以序列化的(具体序列化的过程都是由.NET内部进行的),那么我们就可以把此类放入ViewState中了,如果没有指定 SerializableAttribute属性的话,放入ViewState时就会报错。

以上所示是序列化的第一种方式: 基本序列化,也是比较简单的一种,如果是复杂情况就要使用下面所说的第二种序列化的方式: 自定义序列化
假设我们的Test类需要从DataTable继承:
[SerializableAttribute]
class  Test : DataTable
{
    
public Test()
    
{
        DataColumn col 
= new DataColumn();
        col.DataType 
= typeof(string);
        col.ColumnName 
= "name";
        
this.Columns.Add(col);
    }


    
public object this[int index]
    
{
        
get
        
{
            
if (index >= Rows.Count)
            
{
                
return null;
            }

            
return Rows[index]["name"];
        }

        
set
        
{
            DataRow row 
= NewRow();
            row[
"name"= value;
            Rows.Add(row);
        }

    }

}
那么再把这个类放入ViewState的话就会报错: 此页的状态信息无效,可能已损坏,主要是因为它的父类DataTable中的DataRow和DataColumn等是不可序列化的,我们就需要把这个类改造成如下形式:
[SerializableAttribute]
class  Test : DataTable, System.Runtime.Serialization.ISerializable
{
    
public Test()
    
{
        DataColumn col 
= new DataColumn();
        col.DataType 
= typeof(string);
        col.ColumnName 
= "name";
        
this.Columns.Add(col);
    }


    
public object this[int index]
    
{
        
get
        
{
            
if (index >= Rows.Count)
            
{
                
return null;
            }

            
return Rows[index]["name"];
        }

        
set
        
{
            DataRow row 
= NewRow();
            row[
"name"= value;
            Rows.Add(row);
        }

    }


    
public Test(SerializationInfo info, StreamingContext context)
    
{
        DataColumn col 
= new DataColumn();
        col.DataType 
= typeof(string);
        col.ColumnName 
= "name";
        
this.Columns.Add(col);

        ArrayList list 
= info.GetValue("list"typeof(ArrayList)) as ArrayList;
        
foreach (string value in list)
        
{
            DataRow row 
= NewRow();
            row[
"name"= value;
            Rows.Add(row);
        }

    }


    
public void GetObjectData(SerializationInfo info, StreamingContext context)
    
{
        ArrayList list 
= new ArrayList();
        
foreach (DataRow row in this.Rows)
        
{
            list.Add(row[
"name"]);
        }
 
        info.AddValue(
"list", list);
    }


}
1、实现 ISerializable接口
2、实现 GetObjectData方法,这个方法中就是把要序列化的对象放入info中,特别注意放入info中的对象本身必须是可以序列化的,如果放入一个DataRow对象,就会报错: 未标记为可序列化
3、实现 public Test(SerializationInfo info, StreamingContext context) 构造函数,这个函数就是一个反序列化的操作,把info中的对象取出来

经过上面的改造之后,就可以把这个Test对象放入ViewState中了



    本文转自永春博客园博客,原文链接:http://www.cnblogs.com/firstyi/archive/2007/11/20/965957.html,如需转载请自行联系原作者


相关文章
|
3月前
|
存储 JSON 安全
解密序列化:背后的魔法与陷阱
解密序列化:背后的魔法与陷阱
28 0
|
JavaScript Java 索引
针对EL表达式的本质以及规范的透析
EL(Expression Language) 是为了使JSP写起来更加简单。表达式语言的灵感来自于 ECMAScript 和 XPath 表达式语言,它提供了在 JSP 中简化表达式的方法,让Jsp的代码更加简化。
针对EL表达式的本质以及规范的透析
关于VB6.0中控件加载的难题
我想大家都多多少少都为VB6.0中ActiveX控件的加载而感到头痛,比如有时候在打开一个工程的瞬间它给你弹出无法加载或加载错误的提示框,在你想用到某个控件时,VB6.0中又找不到……这样的情况真的很让人恼火!!!
|
存储 XML 安全
Java序列化技术即将被废除!!!
我们的对象并不只是存在内存中,还需要传输网络,或者保存起来下次再加载出来用,所以需要Java序列化技术。Java序列化技术正是将对象转变成一串由二进制字节组成的数组,可以通过将二进制数据保存到磁盘或者传输网络,磁盘或者网络接收者可以在对象的属类的模板上来反序列化类的对象,达到对象持久化的目的。
101 0
|
前端开发 安全 程序员
详解 WWDC 20 SwiftUI 的重大改变及核心优势
随着 WWDC 20 相关新特性和介绍视频的释出,都明确的宣告着 SwiftUI 元年已经到了,SwiftUI 已经成长为新时代的布局引擎。 以下从几个方面分享关于 SwiftUI 的重大改变及核心优势。
2867 0
详解 WWDC 20 SwiftUI 的重大改变及核心优势
|
Java 数据库
再生与终结-初识属性覆盖与final | 带你学《Java面向对象编程》之四十一
本节将为读者介绍属性覆盖和final关键字相关内容,并为读者展示如何在Java中定义一个“常量”。
|
Java C++
保守VS开放?看清封装对象属性 | 带你学《Java面向对象编程》之四
高楼万丈,起于平地。本节通过对比正反几个实例剖析了封装对象属性的必要性,介绍了进行封装的基本原则。
保守VS开放?看清封装对象属性   |  带你学《Java面向对象编程》之四