ASP.Net中的新狀態容器
前面我們提到,ASP.Net為保存用戶請求間的數據添加了幾種新的途徑。這些途徑給了你如何保持狀態信息更好的控制。這些技術的范圍可以窄到只有一個請求那麼小(Context對象),也可以寬到整個Web服務器和服務器上的所有應用程序(Machine.config文件)。在多數情況下你有多種保存特定數據片的選擇--使用每個方法描述的問題和答案來決定某個對象是否適合你的需要。
Cache
Cache對象用於單個用戶、一組用戶或所有的用戶。這種數據為多個請求保持。它可以保持很長時間,但是不能超過應用程序重新啟動的時間,並且數據的終止基於時間或者其它的依賴關系。它可以高效率地保持大量或少量地數據。
Cache 是ASP.NET中最"酷"的對象之一。它提供了難以置信的靈活性、通用性和性能,因此在ASP.Net應用程序中它通常是比Application或Sessions更好的保持數據的對象。本文沒有詳細介紹Cache對象的使用方法,但是仍然可以說它是一個萬能對象。與其它的集合對象相似,它是一個簡單的名稱-值集合,但是通過使用指定特定用戶的鍵值可以緩存特定用戶的值。同樣你可以緩存不同的相關數據的多個數據集,例如幾個有鍵(如fordcars 、 chevycars、gmcars)的汽車集合。Cache中的數據可以給定一個絕對的、可變的或基於文件的終止時間。它們也實現了一個回調功能,在被緩存的值從緩存中提取時被調用,這個功能很有用,因為接著你能檢查它是否為最新的數據變量,如果不是(或數據源不可用),就重新緩存被終止的值。
添加和訪問緩存中值的語法與先前談到的相似。但是Cache給訪問集合內容的標准索引器方法作了補充,它支持多種方法,允許對被緩存數據的更多的控制。最頻繁使用的方法是Insert,它支持幾種重載,允許你指定依賴、超時值、優先級和回調。下面是一些簡單的例子:
// 給緩存添加項
Cache["myKey"] = myValue;
// 從緩存中讀取項
Response.Write(Cache["myKey"]);
// 把CacheDuration增加10秒並把項添加到緩存中
Cache.Insert("myKey",myValue, null, System.DateTime.Now.AddSeconds(10),
System.Web.Caching.Cache.NoSlidingExpiration);
Cache對象的最強大的特性之一是當緩存中的某個項終止時執行回調的能力。它使用了委托或函數指針,這在本文中沒有討論。幸運的是一旦你有了某些這些技術怎樣工作的示例,就能通過簡單的剪切和粘貼在應用程序中使用它們,不需要知道委托是怎樣工作的復雜過程。有很多使用這種功能的原因,最通常的是在數據終止時用當前數據重新填充緩存,或者如果重新填充緩存的數據源不可用時恢復舊的緩存數據。
在我的例子中,簡單地緩存了當前時間,當緩存超期的時候,我將給緩存中的字符串末尾添加一個星號(*)。在超過時間後,你能通過計算星號的數量來確定緩存超期了多少次。圖9演示了回調的重要概念,並且提供了給使用緩存建立更多功能回調程序的好模板。
private void Page_Load(object sender, System.EventArgs e)
{
string cacheKey = "myKey";
string data = "";
// 檢查數據是否已經被緩存了
if(Cache[cacheKey]==null)
{
// 因為數據在緩存中,所有讀取數據
data = System.DateTime.Now.ToString();
//建立回調委托的一個實例
CacheItemRemovedCallback callBack =new CacheItemRemovedCallback(onRemove);
Label1.Text = "Generated: " + data;
Cache.Insert(cacheKey,data,null,
System.DateTime.Now.AddSeconds(5),
System.Web.Caching.Cache.NoSlidingExpiration,
System.Web.Caching.CacheItemPriority.Default,
callBack);
}
else
{
Label1.Text = "Cached: " + Cache[cacheKey].ToString();
}
}
private void onRemove(string key, object val,CacheItemRemovedReason reason)
{
//建立回調委托的一個實例
CacheItemRemovedCallback callBack =new CacheItemRemovedCallback(onRemove);
Cache.Insert(key,val.ToString() +
"*",null,System.DateTime.Now.AddSeconds(5),Cache.NoSlidingExpiration,
System.Web.Caching.CacheItemPriority.Default, callBack);
}
代碼段5.緩存回調示例
注意代碼段中一個重要的特性是在Page_Load中使用模式(pattern)來確定是否使用緩存中的數據。當你處理緩存中的項時也可能使用這種模式。使用if語句來檢查緩存的當前內容是否為空(因為要多次引用,為緩存鍵使用了一個變量)。如果是空的,從數據源生成數據並放入緩存中。如果不是空的,從緩存中返回數據。如果數據訪問邏輯很復雜,你需要把整個if語句放入一個獨立的函數,該函數的任務是檢索數據。
Cache對象的功能比先前我們討論的大多數對象多得多。這也是ASP.Net更強大的功能之一,並且我明確地推薦閱讀關於它的更多內容。
Context
Context對象保持單個用戶、單個請求的數據,並且數據只在該請求期間保持。Context容器可以保持大量的數據,但是典型的情況下是保存小的數據片,因為它經常通過global.asax中的某個處理方法為每個請求實現。
Context容器(從Page對象訪問或使用System.Web.HttpContext.Current)被提供用於保持需要在不同的HttpModules和HttpHandlers之間傳遞的值。它也可以用於保持某個完整請求的相應信息。例如,IbuySpy入口在global.asax中的Application_BeginRequest事件過程中給容器填滿了許多配置信息。注意這只在當前請求中可用,如果你希望在下一個請求中也能使用,請考慮使用VIEwState。
從Context集合中設置和獲取數據使用的語法與前面討論的其它集合對象(如Application、Sessions和 Cache)的相似。下面是兩個簡單的例子:
// 給Context添加項
Context.Items["myKey"] = myValue;
// 從Context中讀取項
Response.Write(Context["myKey"]);
VIEwState
VIEwState為單個用戶保持狀態信息,保持期為ASPX頁面工作時間。ViewState容器可以保持大量的數據,但是必須小心管理VIEwState的大小,因為它增加了每個請求和回應的下載(download)大小。
VIEwState是ASP.Net中的一個新容器,也許你已經使用它了,但是你可能還是不了解它。這是因為所有的內建Web控件都使用ViewState在頁面回發(postback)間保持自己的值。但是你必須小心,因為它影響應用程序的性能。影響的大小依賴於回發之間使用VIEwState的多少--對大多數Web窗體來說數量非常小。
確定某個頁面上每個控件使用的ViewState的數量最簡單的方法是打開頁面追蹤並檢查每個控件負載了多少個ViewState。如果某個特定控件不需要在回發之間保持數據,請通過把EnableViewState設置為false關閉該對象的VIEwState。你也可以通過在浏覽器中查看的Html源並檢查隱藏窗體字段__VIEWSTATE來確定某個給定的ASP.Net頁面ViewState的總共大小。注意這些內容都是使用Base64編碼的,用於放置偶然的查看和維護。ViewState也可以通過給@Page指令添加EnableVIEwState="false"在整個頁面中禁止。
典型的Web窗體不需要直接維護VIEwState。但是如果你建立自定義Web控件,就需要了解它是怎樣工作的,並為你的控件實現它,這樣該控件的工作方式才能與隨ASP.Net發布的Web控件同樣地工作。向VIEwState讀取或寫入值都可以通過上面討論地其它集合對象的語法完成:
// 給VIEwState添加項
VIEwState["myKey"] = myValue;
//從Context讀取項
Response.Write(VIEwState["myKey"]);
當建立自定義Web控件時,你也許希望它們有ViewState的好處。這在控件的屬性層可以簡單實現。代碼段6演示了怎樣保存一個簡單的自定義控件的PersonName屬性到VIEwState中,並在該控件的Render方法中使用它。
namespace MSDN.StateManagement
{
public class HelloPerson : System.Web.UI.Control
{
public string PersonName
{
get
{
string s = (string)VIEwState["PersonName"];
return ((s == null) ? "" : s);
}
set
{
VIEwState["PersonName"] = value;
}
}
protected override void Render(System.Web.UI.HtmlTextWriter writer)
{
writer.Write("Hello " + PersonName);
}
}
}
代碼段6.在VIEwState中保存數據
Web.config和Machine.config文件
這些文件中的數據對於某個應用程序的所有用戶來說都可以使用。Web.config文件中存儲的數據可用於應用程序的整個生命周期。這些數據一般很小,該對象一般用於保持文件位置和數據庫連接的字符串。大的數據片最好保存在其它位置。
作為其它多樣集合對象的補充,ASP.NET引入了一組XML配置文件用於管理應用程序甚至於整個服務器的很多設置。每個ASP.Net應用程序使用Web.config文件來設置它的許多屬性,每個服務器在系統文件夾下有一個作為應用程序基礎的Machine.config文件。這些設置都作為默認值使用,除非重載。作為保存配置數據的補充,這些文件可以保存應用程序(或多個應用程序)需要的數據。
無論什麼時候應用程序啟動都會讀取配置信息,接著這些信息被緩沖。由於被緩沖了,應用程序可以快速讀取它們,因此不需要考慮應用程序的瓶頸,因為它經常執行某個文本文件的一些整型信息。此外,某個應用程序的Web.config的改變將導致應用程序重新啟動。這確保了對配置文件信息的修改立即反映到應用程序中。
數據庫連接信息,默認圖像路徑和XML數據文件路徑是通常保存在Web.config文件中的數據片。在Web.config文件中保存數據的語法如下,在理想的情況下你也許希望使用集成的SQL身分驗證:
<configuration>
<!-應用程序特殊設置 -->
<aPPSettings>
<add key="connectionString" value="server=myDBServer;
uid=myUID;pwd=myPassWord;database=myDB" />
</aPPSettings>
<system.web>
<!-所有的wsb設置 -->
</system.web>
</configuration>
為了訪問ASP.Net頁面中的值,可以使用ConfigurationSettings集合,它在System.Configurat
ion名字空間中。下面的簡單例子演示了怎樣提取前面的連接字符串到一個本地變量中:
using System.Configuration;
ooo
String strConnString =
ConfigurationSettings.APPSettings["connectionString"];
給System.Configuration名字空間添加一個引用減少了引用這些值的代碼數量。因為對Web.config或 Machine.config的修改將導致應用程序立即重新啟動,典型情況下這些值只由服務器系統管理員手動修改。因此你可以認為這些文件是保存只讀數據而不是應用程序中修改的數據的好位置。
結論
有效的狀態管理意味著識別的用戶經驗、數據錯誤與快速的頁面或事務處理之間的巨大差別。盡管狀態管理在ASP 3.0中不太適用,但是ASP.Net把它帶到了本文討論的狀態對象的控制之下。小心地使用它們將使你給用戶展示最佳的Web經驗。