ASP.NET状态管理
Http協(xié)議————“無狀態(tài)協(xié)議”
Web服務(wù)器每分鐘對上千個用戶進(jìn)行管理的一種方式就是執(zhí)行所謂的“無狀態(tài)”鏈接。只要有一個希望瀏覽器返回一個頁面、圖像或其他資源的請求,就發(fā)生以下事情:
1、連接到服務(wù)器
2、告訴服務(wù)器想要的頁面、圖像或者其他項
3、服務(wù)器發(fā)送請求的資源
4、服務(wù)器切斷連接,把用戶忘的干干凈凈。
也就是頁面之間在Http協(xié)議下是沒有任何關(guān)系的,這樣就需要有狀態(tài)管理來傳輸頁面之間的數(shù)據(jù)。
WEB頁面處理過程
1、頁面的一次往返處理:用戶對服務(wù)器控件的一次操作,就可能引起頁面的一次往返處理:頁面被提交到服務(wù)器端,執(zhí)行響應(yīng)的事件處理代碼,重建頁面,然后返回到客戶端。
2、頁面重建:每一次頁面被請求,或者頁面事件被提交到服務(wù)器,ASP.NET運(yùn)行環(huán)境將執(zhí)行必要的代碼,重建整個頁面,把結(jié)果頁面送到瀏覽器,然后拋棄頁面的變量、控件的狀態(tài)和屬性等等頁面信息。
3、頁面處理內(nèi)部過程:
?? (1)、Page_Load:IsPostBack屬性判定頁面是否為第一次被請求。
?? (2)、事件處理:這一階段處理表單的事件
?? (3)、Page_Unload:這個階段頁面已經(jīng)處理完畢,需要做些清理工作,一般地,你可以在這個階段關(guān)閉打開文件和數(shù)據(jù)庫鏈路,或者釋放對象。
ASP.NET Web Form框架的“連續(xù)”和“有狀態(tài)”假象
訪問者打開頁面的連續(xù)畫面,這實際上一種假象,這種假象是由ASP.NET頁框架、頁及其控件實現(xiàn)的。
第二部分:基于客戶端的狀態(tài)管理:
狀態(tài)信息放在客戶端上。
一、視圖狀態(tài)(ASP.NET特有)
二、隱藏的窗體域
三、Cookie
四、查詢字符串
一、視圖狀態(tài)
ASP.ENT使用了ViewState視圖狀態(tài),是所有服務(wù)器控件的一個屬性。如果你查看Web Form產(chǎn)生的HTML代碼,可以看到一個名為_ViewState的隱藏字段,ASP.NET將狀態(tài)信息以Hash的方式存儲在這里,通過它,可以在下一次回發(fā)時知道回發(fā)前各控件的狀態(tài)。
ASP.NET服務(wù)器控件的生命周期:
1、初始化——Init事件(OnInit方法)
2、加載視圖狀態(tài)——LoadViewState方法(當(dāng)頁面回發(fā)時,首先從_ViewSatae字段信息中加載該控件的狀態(tài))
3、處理回發(fā)數(shù)據(jù)——LoadPostData方法
4、加載——Load事件(OnLoad方法)
5、發(fā)送回發(fā)更改通知——RaisePostDataChangedEvent方法
6、處理回發(fā)事件——RaisePostBackEvent方法
7、預(yù)呈現(xiàn)——PreRender事件(OnPreRender方法)
8、保存視圖狀態(tài)——SaveViewState方法
9、呈現(xiàn)——Render方法
10、處置——Dispose方法
11、卸載——UnLoad事件(OnUnLoad方法)
視圖狀態(tài)的用法:
1、啟用視圖狀態(tài),即EnableViewState = "true",默認(rèn)為true,如果為false,那么該控件和子控件的視圖狀態(tài)就不會被串行化。
2、可以在視圖狀態(tài)中存儲的類型:Int32,bool,string,color,array,arrayList,Unit以及以上類型的HashTable對象
3、視圖狀態(tài)與安全
視圖狀態(tài)串行化的字符串表達(dá)式作為明文來往返傳送,這是不安全的,在視圖狀態(tài)中絕不能保存任何信息(例如口令、連接字符、文件路徑等。)
參考例子(ViewState.aspx和ViewState_userinfo.aspx)
using?System.Data;
using?System.Configuration;
using?System.Collections;
using?System.Web;
using?System.Web.Security;
using?System.Web.UI;
using?System.Web.UI.WebControls;
using?System.Web.UI.WebControls.WebParts;
using?System.Web.UI.HtmlControls;
namespace?ASPNETState
{
????public?partial?class?ViewState?:?System.Web.UI.Page
????{
????????protected?void?Page_Load(object?sender,?EventArgs?e)
????????{
????????????if?(!IsPostBack)
????????????{
????????????????Label1.Text?=?"EnableViewState=True";
????????????????//EnableViewState設(shè)為True時,Lable2的狀態(tài)保存在_viewState隱藏字段中,當(dāng)刷新頁面時,不執(zhí)行此語句,但是可以從_ViewState中取到以前的狀態(tài)信息,所以還是原值。
????????????????Label2.Text?=?"EnableViewState=False";???
????????????????//EnableViewState設(shè)為False時,Lable2的狀態(tài)沒有保存在_viewState隱藏字段中,當(dāng)刷新頁面時,不執(zhí)行此語句,并且取不到以前的狀態(tài)信息,所以為空。
????????????}
????????}
????????protected?void?Button2_Click(object?sender,?EventArgs?e)
????????{
????????????if?(ViewState["count"]?==?null) //從ViewState中讀數(shù)據(jù)
????????????{
????????????????ViewState["count"]?=?0; //向ViewState中保存數(shù)據(jù)(可以保存多種類型的數(shù)據(jù))
????????????}
????????????int?ncount?=?(Int32)ViewState["count"];
????????????ncount?+=?1;
????????????ViewState["count"]?=?ncount;
????????????Label3.Text?=?ViewState["count"].ToString();
????????}
????????protected?void?Button1_Click(object?sender,?EventArgs?e)
????????{
????????}
????}
}
?
二、隱藏的窗體域
1、隱藏域不會顯示在用戶的瀏覽器中,但我們可以像設(shè)置標(biāo)準(zhǔn)控制的屬性那樣設(shè)置其屬性。當(dāng)一個網(wǎng)頁被提交給服務(wù)器時,隱藏域的內(nèi)容和其他控制的值一塊兒被送到HTTP Form集合中。隱藏域可以是任何存儲在網(wǎng)頁中的與網(wǎng)頁有關(guān)的信息的存儲庫,隱藏域在其value屬性中存儲一個變量,而且必須被顯性地添加在網(wǎng)頁上。
2、ASP.NET中的HtmlInputHidden控制提供了隱藏域的功能。
簡單的說,就是頁面上有一個保存值的隱藏控件,頁面上可以取到它的值,但是看不見它。
例子見(hidden.aspx)
三、Cookie
Cookie定義:
1、由網(wǎng)絡(luò)服務(wù)器發(fā)送出來以儲存在網(wǎng)絡(luò)瀏覽器上小量信息;
2、Cookie是把與用戶和網(wǎng)站相關(guān)的信息存儲比會話時間還長的一種方式。
3、Cookie存儲在用戶的硬盤上(一般存儲在Web瀏覽器軟件所在的文件夾上,稱為Cookies)
用途:
1、用戶的個人配置
2、注冊和“Remember me”
3、彈出窗口
如何工作:
1、Cookie存儲在C:\Documents and Settings\<Username>\Cookies下
2、IE選項中的“隱私”選項下修改cookie設(shè)置,也可以在“常規(guī)”選項卡下選擇“刪除cookie”
如何使用:
1、使用Response對象設(shè)置Cookie狀態(tài)
?? Response.Cookies["UserName"].Value = "張三";
2、使用Request對象讀取已有的Cookie
?? string strName = Request.Cookies["UserName"].Value;
3、清除:
?? Response.Cookies["UserName"].Value = null;或
?? Response.Cookies["UserName"].Expires = new System.DateTime(1999,10,12);(過期)
屬性:
1、Value:值,是string類型的
2、Domain:設(shè)置這個屬性后,只有在這個域下才能訪問該Cookie
? 例如:
? Response.Cookies["UserName"].Domain = ".Webcast.com.cn";
? //只有指定以“.Webcast.com.cn”結(jié)尾的域可以訪問本Cookie
3、Path:該屬性指定那些路徑下的頁面可以訪問此Cookie
4、Expires:指定Cookie過期的日期(清除Cookie)
? Response.Cookies["UserName"].Expires = new System.DateTime(1999,10,12);(過期)
參見實例(Cookie.aspx)
該例子是說在頁面上的TextBox中輸入一個用戶名,然后點“注冊”按鈕,此時將用戶名存入Cookie中,然后在頁面加載時取Cookie值顯示在頁面中,同時10s中自動刪除Cookie。
using?System.Data;
using?System.Configuration;
using?System.Collections;
using?System.Web;
using?System.Web.Security;
using?System.Web.UI;
using?System.Web.UI.WebControls;
using?System.Web.UI.WebControls.WebParts;
using?System.Web.UI.HtmlControls;
namespace?ASPNETState
{
????public?partial?class?Cookie?:?System.Web.UI.Page
????{
????????protected?void?Page_Load(object?sender,?EventArgs?e)
????????{
????????????if?(Request.Cookies["UserName"]?!=?null) //注意是Request
????????????{
????????????????Response.Write("親愛的"?+?Request.Cookies["UserName"].Value?+?",歡迎你光臨!");
????????????????
????????????}
????????????else
????????????{
????????????????Response.Write("歡迎光臨!");
????????????}
????????????
????????}
????????protected?void?btnZC_Click(object?sender,?EventArgs?e)
????????{
????????????Response.Cookies["UserName"].Value?=?txtUser.Text;
????????????Response.Cookies["UserName"].Expires?=?DateTime.Now.AddSeconds(10);??//10秒鐘后失效,即Cookies["UserName"]的信息丟失;或者手工刪除(IE中選項刪除Cookie)
????????}
????}
} 四、查詢字符串
在講查詢字符串之前先了解一下GET 和 POST方法的不同:
?? 通過HTTP從Web服務(wù)器請求頁面或其他資源,有兩個通用的方法(GET 和 POST方法)。可使用GET方法直接獲得資源,也可使用POST把值傳給相應(yīng)資源。GET方法是缺省的
?? 假如把一個或多個成對的名稱/值附在請求頁面的URL后,就變成請求的查詢字符串,且在QueryString集合中提供給ASP頁面。單擊Web頁面、Email消息或其它文檔的超鏈接,或在瀏覽器的地址欄中輸入地址并按回車,或單擊瀏覽器中的Links或Favorites按鈕,所有這些都要使用GET方法。
?? 因此,對這些動作中傳遞值給ASP的唯一方法是通過QueryString集合,把值附在URL后。
?? http://mysite.com/process_page.asp?FirstName=Priscilla&LastName=Descartes
?? 可以采用如下方式訪問在QueryString集合中提供的值(QueryString就是下面要講的查詢字符串):
?? strFirstName = Request.QueryString("FirstName")?? ''Return "Priscilla"
?? strLastName = Request.QueryString("LastName")???? ''Return "Descartes"
?? strRaw = Request.QueryString?
?? Return "FirstName=Priscilla&LastName=Descartes"
在一個頁面內(nèi)使用<FORM>段時,可以設(shè)置打開的FORM標(biāo)記的METHOD屬性值為“GET”或“POST”,缺省值為“GET”。假如使用“GET”或省略其屬性,瀏覽器將該值綁定在頁面所有控件上,成為一個查詢字符串,且附在被請求頁面的URL上。當(dāng)這個請求到達(dá)Web服務(wù)器時,其值由ASP的Request.QueryString集合提供。然而,假如設(shè)置METHOD屬性為“POST”,瀏覽器將值包裝進(jìn)發(fā)送服務(wù)器的HTTP報頭中,通過Request.Form集合提供給ASP。
?? 通常來說,可以在所有的HTML窗體中使用GET方法。然而,瀏覽器或服務(wù)器的URL字符串長度存在一定的限制。因此,附有長的字符串可能會引起溢出和某些字符串的字符被截掉。同時,查詢字符串出現(xiàn)在瀏覽器的地址欄和所有的保存的鏈接和收藏夾中。不僅如此,還顯露了通過Web服務(wù)器時在HTTP請求中不想顯示的值,它也可能出現(xiàn)你的服務(wù)器和其他路由服務(wù)器的日志文件中。在HTTP請求報頭中的值很少是可見的,并且不出現(xiàn)在日志文件中。
?? 使用POST方法需要注意的小問題是,當(dāng)用戶重新下載<FORM>時,窗體的值將不再保留,其值為空且必須重新輸入。然而,當(dāng)附在URL上時,其值被存儲為一個鏈接,將被保留,因此將出現(xiàn)在所有的URL與字符串結(jié)合的請求中,這或許是個優(yōu)點也可能是個缺點,這根據(jù)應(yīng)用而定(一些瀏覽器在客戶端上能夠在一定范圍內(nèi)自動保留一個頁面上的值)。
先看看get方法(利用查詢字符串):
1、查詢字符串提供了一種簡單而受限制的維護(hù)狀態(tài)信息的方法,我們可以方便地給那個信息從一個網(wǎng)頁傳遞給另一個網(wǎng)頁。
2、帶有查詢字符串的URL如下所示:
http://localhost:1305/QueryString_Show.aspx?username=xieex&password=1111
3、使用:
string sUserName,sPwd;
sUserName = Request.Params["username"].ToString();
sPwd = Request.Params["password"].ToString();
或
sUserName = Request.QueryString["username"].ToString();
sPwd = Request.QueryString["password"].ToString();
或
sUserName = Request["username"].ToString();
sPwd = Request["password"].ToString();
都可以取到xieex和1111
其實這樣的傳值方式很常見也很有用,例如有些系統(tǒng)中,Gird中有很多條數(shù)據(jù),我可以對某條數(shù)據(jù)進(jìn)行瀏覽或者編輯,當(dāng)點“瀏覽”按鈕時,我彈出對話框?qū)υ摋l記錄進(jìn)行瀏覽,此時不允許編輯,此時我就需要從主頁面上傳一個狀態(tài)(state)到對話框頁面上來,然后在對話框頁面上取其狀態(tài)(Request["state"].ToString()),此時根據(jù)其值(edit或browse)就可以控制是否可以編輯了。
參見實例(QueryString.aspx和QueryString_Show.aspx)
該例子是說在QueryString.aspx頁面上注冊用戶名和密碼,然后跳轉(zhuǎn)到另一個頁面上,在另一個頁面上取其用戶名和密碼。
在該頁面上注冊用戶名和密碼
using?System;
using?System.Data;
using?System.Configuration;
using?System.Collections;
using?System.Web;
using?System.Web.Security;
using?System.Web.UI;
using?System.Web.UI.WebControls;
using?System.Web.UI.WebControls.WebParts;
using?System.Web.UI.HtmlControls;
namespace?ASPNETState
{
????public?partial?class?QueryString?:?System.Web.UI.Page
????{
????????protected?void?Page_Load(object?sender,?EventArgs?e)
????????{
????????}
????????protected?void?Button1_Click(object?sender,?EventArgs?e)
????????{
????????????Response.Redirect("QueryString_Show.aspx?username="?+?txtUser.Text?+?"&password="?+?txtPwd.Text);
????????}
????}
}
在另一個頁面上取用戶名和密碼:
using?System;using?System.Data;
using?System.Configuration;
using?System.Collections;
using?System.Web;
using?System.Web.Security;
using?System.Web.UI;
using?System.Web.UI.WebControls;
using?System.Web.UI.WebControls.WebParts;
using?System.Web.UI.HtmlControls;
namespace?ASPNETState
{
????public?partial?class?QueryString_Show?:?System.Web.UI.Page
????{
????????protected?void?Page_Load(object?sender,?EventArgs?e)
????????{
????????????Response.Write("用戶:"?+?Request.Params["username"]+"<br>");
????????????Response.Write("密碼:"?+?Request.Params["password"]+"<br>");
????????????Response.Write("用戶:"?+?Request.QueryString["username"]+"<br>");//get方法時用Request.QueryString
????????????Response.Write("密碼:"?+?Request.QueryString["password"]+"<br>");
????????????Response.Write("用戶:"?+?Request["username"]+"<br>");
????????????Response.Write("密碼:"?+?Request["password"]+"<br>");
????????}
????}
}
?
再看看post方法
直接看例子:(Post.aspx和post_acc.aspx)
該例子和上面例子差不多,是說在Post.aspx頁面上注冊用戶名和密碼,然后跳轉(zhuǎn)到另一個頁面上,在另一個頁面上取其用戶名和密碼。
<!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>無標(biāo)題頁</title>
</head>
<body>
?????<!--注意action="post_acc.aspx"?method="post"-->
????<form?action="post_acc.aspx"?method="post">
????<div>
????????用戶:<input?id="txtUser"?type="text"?name="username"?/><br?/>
????????<br?/>
????????密碼:<input?id="txtPwd"?type="text"?name="pwd"?/><br?/>
????????<br?/>
????????<input?id="Button1"?type="submit"?value="提交"??/></div>
????????<!--注意按鈕的type為submit-->
????</form>
</body>
</html>
在另一個頁面上取用戶名和密碼:
using?System.Data;
using?System.Configuration;
using?System.Collections;
using?System.Web;
using?System.Web.Security;
using?System.Web.UI;
using?System.Web.UI.WebControls;
using?System.Web.UI.WebControls.WebParts;
using?System.Web.UI.HtmlControls;
namespace?ASPNETState
{
????public?partial?class?post_acc?:?System.Web.UI.Page
????{
????????protected?void?Page_Load(object?sender,?EventArgs?e)
????????{
????????????if?(!IsPostBack)
????????????{
????????????????Response.Write(Server.MapPath("post.aspx"));
????????????????Response.Write("用戶:" + Request.Form["username"]);//post方法時用Request.Form
????????????????Response.Write("密碼:"? +?Request.Params["pwd"]);
????????????}
????????}
????}
}
第三部分 基于服務(wù)器的狀態(tài)管理
信息存儲在服務(wù)器上,盡管其安全型較高,但會占用較多的web服務(wù)器資源。服務(wù)器端通常用以下方式實現(xiàn)狀態(tài)管理:
一、Application對象
二、Session對象
1、應(yīng)用程序級別的狀態(tài)存取(就是說服務(wù)器上的應(yīng)用程序,各個客戶端都可以訪問它)
2、變量狀態(tài)的存儲和提取
?? 存儲: Application["username"] = "xieex";
?? 提取: string strUserName = Application["username"];
3、同時訪問要加鎖,防止并發(fā)沖突
?? Application.Lock();
?? Application.Unlock();
Application是保存在服務(wù)器內(nèi)存中的。
參見實例(ApplicationState.aspx)
該實例用于記錄訪問該頁面的訪問者個數(shù),用Application存儲變量,這樣不會因為一次會話結(jié)束而把訪問記錄清零。
using?System;
using?System.Data;
using?System.Configuration;
using?System.Collections;
using?System.Web;
using?System.Web.Security;
using?System.Web.UI;
using?System.Web.UI.WebControls;
using?System.Web.UI.WebControls.WebParts;
using?System.Web.UI.HtmlControls;
namespace?ASPNETState
{
????public?partial?class?ApplicationState?:?System.Web.UI.Page
????{
????????protected?void?Page_Load(object?sender,?EventArgs?e)
????????{
????????????Application.Lock();??//加鎖,防止并發(fā),保證同一時間只有一個用戶對其訪問
????????????if?(Application["count"]?!=?null)
????????????{
????????????????Application["count"]?=?(Int32)Application["count"]?+?1;
????????????}
????????????else
????????????{
????????????????Application["count"]?=?1;
????????????}
????????????Application.UnLock();
????????????Response.Write("您是第"+Application["count"]+"位訪問者!");??
???????????//每訪問一次都加1,只有當(dāng)IIS服務(wù)重啟時才會清零
????????}
????}
}
另一個例子(模擬網(wǎng)站的當(dāng)前用戶人數(shù)和訪問總?cè)藬?shù))
Global.asax
using?System.Data;
using?System.Configuration;
using?System.Collections;
using?System.Web;
using?System.Web.Security;
using?System.Web.SessionState;
using?System.IO;
namespace?ASPNETState
{
????public?class?Global?:?System.Web.HttpApplication
????{
????????/**////?<summary>
????????///?必需的設(shè)計器變量。
????????///?</summary>
????????private?System.ComponentModel.IContainer?components?=?null;
????????private?FileStream?fileStream;
????????private?StreamReader?reader;//讀字符流
????????private?StreamWriter?writer;//寫字符流
????????public?Global()
????????{
????????????InitializeComponent();
????????}????
????????protected?void?Application_Start(object?sender,?EventArgs?e)
????????{
????????????Application["CurrentGuests"]?=?0;//初始花為0;
????????????fileStream?=?File.Open(Server.MapPath("counts.text"),?FileMode.OpenOrCreate);//文件不存在,創(chuàng)建文件
????????????reader?=?new?StreamReader(fileStream);//要讀取的完整路徑
????????????Application["AllGuests"]?=?Convert.ToInt32(reader.ReadLine());?//從當(dāng)前流中讀取一行字符并將數(shù)據(jù)作為字符串返回
????????????reader.Close();//關(guān)閉流
????????}
????????protected?void?Session_Start(Object?sender,?EventArgs?e)//當(dāng)用戶訪問網(wǎng)站時,在線用戶+1,總訪問數(shù)+1
????????{
????????????Application.Lock();//同步,避免同時寫入
????????????Application["CurrentGuests"]?=?(int)Application["CurrentGuests"]?+?1;//總在線用戶數(shù)
????????????Application["AllGuests"]?=?(int)Application["AllGuests"]?+?1;//訪問網(wǎng)站的總用戶數(shù)
????????????fileStream?=?new?FileStream(Server.MapPath("counts.text"),?FileMode.OpenOrCreate,?FileAccess.ReadWrite);//
????????????writer?=?new?StreamWriter(fileStream);//實現(xiàn)一個寫入流,使其以一種特定的編碼向流中寫入字符
????????????writer.WriteLine(Application["AllGuests"].ToString());//把訪問網(wǎng)站的總用戶數(shù)再次寫入到文件
????????????writer.Close();//關(guān)閉寫入流
????????????Application.UnLock();//同步結(jié)束
????????}
????????protected?void?Application_BeginRequest(Object?sender,?EventArgs?e)
????????{
????????}
????????protected?void?Application_EndRequest(Object?sender,?EventArgs?e)
????????{
????????}
????????protected?void?Application_AuthenticateRequest(Object?sender,?EventArgs?e)
????????{
????????}
????????protected?void?Application_Error(Object?sender,?EventArgs?e)
????????{
????????}
????????protected?void?Session_End(Object?sender,?EventArgs?e)//當(dāng)前用戶退出網(wǎng)站時,在線用戶數(shù)量-1,
????????{
????????????Application.Lock();
????????????Application["CurrentGuests"]?=?(int)Application["CurrentGuests"]?-?1;//總在線用戶數(shù)量-1
????????????Application.UnLock();
????????}
????????protected?void?Application_End(Object?sender,?EventArgs?e)
????????{
????????}
????????????
????????private?void?InitializeComponent()
????????{
????????????this.components?=?new?System.ComponentModel.Container();
????????}
????}
}
在頁面上顯示,需寫代碼:
????????{
????????????this.Label1.Text?=?"正在訪問站點的用戶數(shù):"?+?Application["CurrentGuests"].ToString();
????????????this.Label2.Text?=?"訪問過站點的總用戶數(shù):"?+?Application["AllGuests"].ToString();?
????????}
Application對象的使用建議:
1、對于頻繁使用(很多用戶都要使用的)的數(shù)據(jù)使用該對象
2、不要把太多的信息放在該對象中
3、如果站點有很大的通信量,建議使用Web.Config
二、Session狀態(tài)
對網(wǎng)站的一次訪問叫做會話(Session),超時后,自動結(jié)束會話(一般是20分鐘),Session也是保存在服務(wù)器內(nèi)存中的。
使用Session時的情況,如:
1、購物車:網(wǎng)絡(luò)用戶決定購買的商品列表
2、用戶信息:訪問者的姓名
3、用戶設(shè)置:個性化界面
等等
ASP.NET會話狀態(tài)模塊在Web.config文件中像這樣配置(不進(jìn)行額外設(shè)置,以下是默認(rèn)設(shè)置)的:
<sessionState mode="InProc" cookieless="false" timeout="20" />
mode屬性設(shè)為InProc(默認(rèn)值),表明會話狀態(tài)要由ASP.NET存儲在內(nèi)存中
cookieless屬性設(shè)為false,表明不用Cookie來傳遞會話ID,這就避免了用戶禁用了Cookie,Session對象無數(shù)據(jù)可用。
timeout屬性設(shè)為20,表示登錄網(wǎng)站后,如果20分鐘不對其進(jìn)行操作,則該會話結(jié)束,需要重新登錄。
Session屬性和方法:
1、TimeOut屬性:
獲取和設(shè)置會話結(jié)束之前的時間段,以分鐘為單位,默認(rèn)為20分鐘
2、Abandon():
結(jié)束當(dāng)前會話,會話中的所有信息都被清空
3、Clear():
刪除當(dāng)前會話中的所有信息,但不結(jié)束會話
4、IsNewSession:
如果會話是在用戶訪問頁面時創(chuàng)建的,則這個屬性返回true,當(dāng)會話需要對某些數(shù)據(jù)進(jìn)行初始化后才能使用時,就可以使用這個屬性
參見實例(SessionState.aspx和SessionState_Redirect.aspx)
該例子是說在SessionState.aspx頁面上注冊用戶名和密碼,然后跳轉(zhuǎn)到另一個頁面上,在另一個頁面上取其用戶名和密碼。
在該頁面上注冊用戶名和密碼
?
using?System;using?System.Data;
using?System.Configuration;
using?System.Collections;
using?System.Web;
using?System.Web.Security;
using?System.Web.UI;
using?System.Web.UI.WebControls;
using?System.Web.UI.WebControls.WebParts;
using?System.Web.UI.HtmlControls;
namespace?ASPNETState
{
????public?partial?class?SessionState?:?System.Web.UI.Page
????{
????????protected?void?Page_Load(object?sender,?EventArgs?e)
????????{
????????}
????????protected?void?Button1_Click(object?sender,?EventArgs?e)
????????{
????????????Session["user"]?=?txtUser.Text;
????????????Session["pwd"]?=?txtPwd.Text;
????????????Response.Redirect("SessionState_Redirect.aspx");
????????}
????}
}
在另一個頁面上取用戶名和密碼:
using?System.Data;
using?System.Configuration;
using?System.Collections;
using?System.Web;
using?System.Web.Security;
using?System.Web.UI;
using?System.Web.UI.WebControls;
using?System.Web.UI.WebControls.WebParts;
using?System.Web.UI.HtmlControls;
namespace?ASPNETState
{
????public?partial?class?SessionState_Redirect?:?System.Web.UI.Page
????{
????????protected?void?Page_Load(object?sender,?EventArgs?e)
????????{
????????????Label1.Text?=?Session["user"].ToString();
????????????Label2.Text?=?Session["pwd"].ToString();
????????}
????}
}
?
Application和Session狀態(tài)的區(qū)別:
簡單的說,Application是應(yīng)用程序級別的狀態(tài)存儲,Session是會話級別的狀態(tài)存儲。
另外作用域不同,
Application對象針對所有用戶都生效
Session對象則相反,每個用戶都有自己的Session對象,它的生命周期起始于服務(wù)器產(chǎn)生對用戶請求頁面的響應(yīng),終止于用戶斷開與服務(wù)器的連接。
舉例來說,當(dāng)手機(jī)開機(jī)時,相當(dāng)于一個Applicaion開始,然后當(dāng)一個朋友打電話過來,此時一個Session開始,掛電話時相當(dāng)于這個Session結(jié)束,然后又有一個朋友打電話過來,此時另一個Session開始。手機(jī)未關(guān)機(jī)說明Application還未結(jié)束
由于Application和Session狀態(tài)都存儲在內(nèi)存中,但是當(dāng)服務(wù)器重新啟動時,保留的狀態(tài)就會消失了,為了保留其狀態(tài),就必須將狀態(tài)保存到數(shù)據(jù)庫。如網(wǎng)站計數(shù)器
最后對各個狀態(tài)進(jìn)行總結(jié):
為了更清楚的了解,我們總結(jié)出每一種對象應(yīng)用的具體環(huán)境,如下表所示:?
| 方法 | 信息量大小 | 保存時間 | 應(yīng)用范圍 | 保存位置 |
| Application | 任意大小 | 整個應(yīng)用程序的生命期 | 所有用戶 | 服務(wù)器端 |
| Session | 小量,簡單的數(shù)據(jù) | 用戶活動時間+一段延遲時間(一般 | 單個用戶 | 服務(wù)器端 |
| Cookie | 小量,簡單的數(shù)據(jù) | 可以根據(jù)需要設(shè)定 | 單個用戶 | 客戶端 |
| Viewstate | 小量,簡單的數(shù)據(jù) | 一個Web頁面的生命期 | 單個用戶 | 客戶端 |
| Cache | 任意大小 | 可以根據(jù)需要設(shè)定 | 所有用戶 | 服務(wù)器端 |
| 隱藏域 | 小量,簡單的數(shù)據(jù) | 一個Web頁面的生命期 | 單個用戶 | 客戶端 |
| 查詢字符串 | 小量,簡單的數(shù)據(jù) | 直到下次頁面跳轉(zhuǎn)請求 | 單個用戶 | 客戶端 |
| Web.Config文件 | 不變或極少改變的小量數(shù)據(jù) | 直到配置文件被更新 | 單個用戶 | 服務(wù)器端 |
1、ViewState對象
??? ViewState 常用于保存單個用戶的狀態(tài)信息,有效期等于頁面的生存期。ViewState容器可以保持大量的數(shù)據(jù),但是必須謹(jǐn)慎使用,因為過多使用會影響應(yīng)用程序的性能。所有Web服務(wù)器控件都使用ViewState在頁面回發(fā)期音保存自己的狀態(tài)信息。如果某個控件不需要在回發(fā)期間保存狀態(tài)信息,最好關(guān)閉該對象的ViewState,避免不必要的資源浪費(fèi)。通過給@Page指令添加“EnableViewState=false”屬性可以禁止整個頁面的ViewState。
2、隱藏域
? Hidden控件是屬于HTML類型的服務(wù)器控件,使用此控件可以實現(xiàn)隱藏域的功能。其實此控件和其它服務(wù)器控件的使用沒有太大區(qū)別,只是它不會在用戶端的瀏覽器中顯示,始終處于隱藏狀態(tài)。但是每次頁面提交的時候,此控件和其它服務(wù)器控件一同提交到服務(wù)器端,因此在服務(wù)器端可以使用Value屬性獲取或保存一些數(shù)據(jù)信息。
3、Cookie對象
??? Cookie用于保存客戶瀏覽器請求服務(wù)器頁面的請求信息,程序員也可以用它存放非敏感性的用戶信息,信息保存的時間可以根據(jù)需要設(shè)置.如果沒有設(shè)置Cookie失效日期,它們僅保存到關(guān)閉瀏覽器程序為止.如果將Cookie對象的Expires屬性設(shè)置為Minvalue,則表示Cookie永遠(yuǎn)不會過期.Cookie存儲的數(shù)據(jù)量很受限制,大多數(shù)瀏覽器支持最大容量為4096,因此不要用來保存數(shù)據(jù)集及其他大量數(shù)據(jù).由于并非所有的瀏覽器都支持Cookie,并且數(shù)據(jù)信息是以明文文本的形式保存在客戶端的計算機(jī)中,因此最好不要保存敏感的,未加密的數(shù)據(jù),否則會影響網(wǎng)站的安全性。
4、查詢字符串
? 查詢字符串的方式是將要傳遞的值連接在URL后面,然后通過Response.Redirect方法實現(xiàn)客戶端的重定向。這種方式可以實現(xiàn)在兩個頁面之間傳遞信息。由于URL的長度有一定的限制,因此不能傳遞太大的信息,加外安全性也不是很好。
5、Application對象
??? Application用于保存所有用戶的公共的數(shù)據(jù)信息,如果使用Application對象,一個需要考慮的問題是任何寫操作都要在Application_OnStart事件(global.asax)中完成.盡管使用Application.Lock和Applicaiton.Unlock方法來避免寫操作的同步,但是它串行化了對Application對象的請求,當(dāng)網(wǎng)站訪問量大的時候會產(chǎn)生嚴(yán)重的性能瓶頸.因此最好不要用此對象保存大的數(shù)據(jù)集合。
6、Session對象
??? Session用于保存每個用戶的專用信息.她的生存期是用戶持續(xù)請求時間再加上一段時間(一般是20分鐘左右).Session中的信息保存在Web服務(wù)器內(nèi)容中,保存的數(shù)據(jù)量可大可小.當(dāng)Session超時或被關(guān)閉時將自動釋放保存的數(shù)據(jù)信息.由于用戶停止使用應(yīng)用程序后它仍然在內(nèi)存中保持一段時間,因此使用Session對象使保存用戶數(shù)據(jù)的方法效率很低.對于小量的數(shù)據(jù),使用Session對象保存還是一個不錯的選擇。
7、Cache對象
?? Cache對象用于在HTTP請求間保存頁面或數(shù)據(jù)。該對象的使用可以極大地提高整個應(yīng)用程序的效率。它允許將頻繁訪問的大量服務(wù)器資源存儲在內(nèi)存中,當(dāng)用戶發(fā)出相同的請求后服務(wù)器不再次處理而是將Cache中保存的信息返回給用戶,節(jié)省了服務(wù)器處理請求的時間。此對象的實例是每個應(yīng)用程序?qū)S玫?#xff0c;其生存期依賴于該應(yīng)用程序的生存期。當(dāng)重新啟動應(yīng)用程序時,將重新創(chuàng)建其Cache對象的實例。使用Cache對象保存信息的代碼如下。?? //存放信息
? Cache["nameID"]="0001";
? //存放信息
? Cache.Insert("nameID","0001"1);
? //讀取信息
? string NameID=Cache["nameID"].ToString();
具體例子代碼見附件:
ASPNETState.rar
總結(jié)
以上是生活随笔為你收集整理的ASP.NET状态管理的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。