asp.net中ViewState的用法详解
在web窗体控件设置为runat = "server",这个控件会被附加一个隐藏的属性_viewstate,_viewstate存放了所有控件在viewstate中的状态值。viewstate是一个名称/值的对象集合。当请求某个页面时,asp.net会把所有控件的状态序列化成一个字符串,然后作为窗体的隐藏属性送到客户端,当客户端吧页面回传时,asp.net分析回传的窗体属性,并赋给控件对应的值。
当我们在写一个asp.net表单时, 一旦标明了 form runat=server ,那么,asp.net就会自动在输出时给页面添加一个隐藏域
<input type="hidden" name="__viewstate" value="">
那么,有了这个隐藏域,页面里其他所有的控件的状态,包括页面本身的一些状态都会保存到这个控件值里面. 每次页面提交时一起提交到后台,asp.net对其中的值进行解码,然后输出时再根据这个值来恢复各个控件的状态. 我们再看这个控件的value值,它可能类似如下的形式:oz4+o2w8atwxpjs+o2w8.... 很多人会认为这是加密的信息,其实不是, ms仅仅是给各个控件和页面的状态存入适当的对象里面,然后把该对象序列化, 最后再做一次base64编码,直接赋值给viewstate控件.
一、viewstate的原理
1.浏览器请求default.aspx页面
2.在服务器端 发现创建的viewstate 这个时候 会自动创建一个名字叫做__viewstate(双下滑线 全部是大写)的隐藏域 其隐藏域的值经过base64加密以后返回到浏览器端这一加密过程在页面生命周期的savestate事件中的saveallstate方法中完成
3.当浏览器提交表单的时候 将__viewstate的隐藏域也一起提交到服务端 这个时候 页面生命周期的readstate事件的readallstate方法会将加密后的值反base64解密 最后将值赋值给名字叫做name的viewstate
4.最后来操作viewstate中的值
二、viewstate的用法:
1.定义viewstate属性
public int pagecount{ get{return (int)viewstate["pagecount"];} set{viewstate["pagecount"]=value;} }
2.使用viewstate的条件
如果要使用viewstate,则在aspx页面中必须要有一个服务器端窗体标记(<form runat = "server">)。窗体字段是必须的,这样包含viewstate信息的隐藏字段才能被传回服务器。而且,该窗体还必须是服务器端的窗体,这样在服务器上执行该页面时,asp.net页面框架才能添加隐藏字段。
page的enableviewstate 属性值为true
控件的enableviewstate 属性值为 true
3.viewstate需要注意的地方
a. 当存在页面回传时,不需要维持控件的值就要把 viewstate 禁止。
b. viewstate的索引是大小写敏感的。
c. viewstate不是跨页面的。
d. 为了能保存在 viewstate中,对象必须是可流化或者定义了 typeconverter。
e. 控件 textbox 的 textmode 属性设置为 password时,它的状态将不会被保存在 viewstate 中,这应该是出于安全性的考虑。
f. 在页面没有回传或重定向或在回传中转到(transfer)其他页面时不要使用 viewstate。
g. 在动态建立控件时要小心它的 viewstate。
h. 当禁止一个程序的 viewstate 时,这个程序的所有页面的 viewstate 也被禁止了。
i. 只有当页面回传自身时viewstate 才是持续的。
4.设置viewstate
viewstate可以在控件,页,程序,全局配置中设置。缺省情况下 enableviewstate 为 true 。如果要禁止所有页面 viewstate 功能,可以在程序配置中把 enableviewstate 设为 false 。
三、产生的 __viewstate如图
使用viewstatedecoder2(viewstate查看器)来看一下值
所以viewstate在安全性上面还是比较差,建议不要存放比较机密和敏感的信息,尽管viewstate可以加密,但是由于viewstate要保存在客户端,天生就有安全性的隐患。
四、viewstate与session的对比
(1) session值是保存在服务器内存上,那么,可以肯定,大量的使用session将导致服务器负担加重. 而viewstate由于只是将数据存入到页面隐藏控件里,不再占用服务器资源,因此, 我们可以将一些需要服务器"记住"的变量和对象保存到viewstate里面. 而sesson则只应该应用在需要跨页面且与每个访问用户相关的变量和对象存储上.
(2) session在默认情况下20分钟就过期,而viewstate则永远不会过期.
但viewstate并不是能存储所有的.net类型数据,它仅仅支持string、integer、boolean、array、arraylist、hashtable 以及自定义的一些类型.
任何事物都有两面性, 使用viewstate会增加页面html的输出量,占用更都的带宽,这一点是需要我们慎重考虑的. 另外, 由于所有的viewstate都是存储在一个隐藏域里面,用户可以很容易的通过查看源码来看到这个经过base64编码的值.然后再经过转换就可以获取你存储其中的对象和变量值。