ViewState的使用比较简单,一两句话就可以了。
赋值:
ViewState[key] = value;
取值:value = ViewState[key];
最主要的作用就是可以在当前页面保存值,ASP.NET的页面状态维护就是使用ViewState来实现的,基本上每一个ASPX页面都可以看到如下类似的html代码:
<input type="hidden" name="__VIEWSTATE" id="__VIEWSTATE" value="/wEPDwUKMTkwNjc4NTIwMWRkyv4ncofW5vaWXdXRtXfXn3RYQR4=" />
data:image/s3,"s3://crabby-images/e95e4/e95e42cc52c789b51b547627ca6c799739e0b9b5" alt=""
也就是说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)
data:image/s3,"s3://crabby-images/9ed40/9ed401c13ef0ca53ee83c3ffe3144daad9d9621b" alt=""
{
Test test = ViewState["VIEW_TEST"] as Test;
if (test == null)
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
test = new Test();
}
test[-1] = TextBox1.Text;
data:image/s3,"s3://crabby-images/0da99/0da994ad2b837f05c4855bad3b115a255fbd7473" alt=""
ViewState["VIEW_TEST"] = test;
}
下面再看一下自定义类Test的实现:
[SerializableAttribute]
class Test
data:image/s3,"s3://crabby-images/9ed40/9ed401c13ef0ca53ee83c3ffe3144daad9d9621b" alt=""
data:image/s3,"s3://crabby-images/849a8/849a86ef3296874633785479796ce82040871888" alt=""
{
private IList list;
data:image/s3,"s3://crabby-images/0da99/0da994ad2b837f05c4855bad3b115a255fbd7473" alt=""
public Test()
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
list = new ArrayList();
}
data:image/s3,"s3://crabby-images/0da99/0da994ad2b837f05c4855bad3b115a255fbd7473" alt=""
public object this[int index]
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
get
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
if (index >= list.Count)
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
return null;
}
return list[index];
}
set
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
list.Add(value);
}
data:image/s3,"s3://crabby-images/0da99/0da994ad2b837f05c4855bad3b115a255fbd7473" alt=""
}
} 特别注意第一行的SerializableAttribute属性,指定这个属性后就代表此类是可以序列化的(具体序列化的过程都是由.NET内部进行的),那么我们就可以把此类放入ViewState中了,如果没有指定SerializableAttribute属性的话,放入ViewState时就会报错。
以上所示是序列化的第一种方式:基本序列化,也是比较简单的一种,如果是复杂情况就要使用下面所说的第二种序列化的方式:自定义序列化
假设我们的Test类需要从DataTable继承:
[SerializableAttribute]
class Test : DataTable
data:image/s3,"s3://crabby-images/9ed40/9ed401c13ef0ca53ee83c3ffe3144daad9d9621b" alt=""
data:image/s3,"s3://crabby-images/849a8/849a86ef3296874633785479796ce82040871888" alt=""
{
public Test()
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
DataColumn col = new DataColumn();
col.DataType = typeof(string);
col.ColumnName = "name";
this.Columns.Add(col);
}
data:image/s3,"s3://crabby-images/0da99/0da994ad2b837f05c4855bad3b115a255fbd7473" alt=""
public object this[int index]
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
get
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
if (index >= Rows.Count)
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
return null;
}
return Rows[index]["name"];
}
set
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
DataRow row = NewRow();
row["name"] = value;
Rows.Add(row);
}
}
} 那么再把这个类放入ViewState的话就会报错:此页的状态信息无效,可能已损坏,主要是因为它的父类DataTable中的DataRow和DataColumn等是不可序列化的,我们就需要把这个类改造成如下形式:
[SerializableAttribute]
class Test : DataTable, System.Runtime.Serialization.ISerializable
data:image/s3,"s3://crabby-images/9ed40/9ed401c13ef0ca53ee83c3ffe3144daad9d9621b" alt=""
data:image/s3,"s3://crabby-images/849a8/849a86ef3296874633785479796ce82040871888" alt=""
{
public Test()
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
DataColumn col = new DataColumn();
col.DataType = typeof(string);
col.ColumnName = "name";
this.Columns.Add(col);
}
data:image/s3,"s3://crabby-images/0da99/0da994ad2b837f05c4855bad3b115a255fbd7473" alt=""
public object this[int index]
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
get
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
if (index >= Rows.Count)
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
return null;
}
return Rows[index]["name"];
}
set
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
DataRow row = NewRow();
row["name"] = value;
Rows.Add(row);
}
}
data:image/s3,"s3://crabby-images/0da99/0da994ad2b837f05c4855bad3b115a255fbd7473" alt=""
public Test(SerializationInfo info, StreamingContext context)
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
DataColumn col = new DataColumn();
col.DataType = typeof(string);
col.ColumnName = "name";
this.Columns.Add(col);
data:image/s3,"s3://crabby-images/0da99/0da994ad2b837f05c4855bad3b115a255fbd7473" alt=""
ArrayList list = info.GetValue("list", typeof(ArrayList)) as ArrayList;
foreach (string value in list)
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
DataRow row = NewRow();
row["name"] = value;
Rows.Add(row);
}
}
data:image/s3,"s3://crabby-images/0da99/0da994ad2b837f05c4855bad3b115a255fbd7473" alt=""
public void GetObjectData(SerializationInfo info, StreamingContext context)
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
ArrayList list = new ArrayList();
foreach (DataRow row in this.Rows)
data:image/s3,"s3://crabby-images/36973/3697370d352d639f06fcffe6068238bbf4bf9202" alt=""
{
list.Add(row["name"]);
}
info.AddValue("list", list);
}
data:image/s3,"s3://crabby-images/0da99/0da994ad2b837f05c4855bad3b115a255fbd7473" alt=""
} 1、实现ISerializable接口
2、实现GetObjectData方法,这个方法中就是把要序列化的对象放入info中,特别注意放入info中的对象本身必须是可以序列化的,如果放入一个DataRow对象,就会报错:未标记为可序列化
3、实现public Test(SerializationInfo info, StreamingContext context) 构造函数,这个函数就是一个反序列化的操作,把info中的对象取出来
经过上面的改造之后,就可以把这个Test对象放入ViewState中了