Cookies想必所有人都了解, 但是未必所有人都精通。本文講解了Cookies的各方面知識, 并且提出來了最佳實踐。這是筆者在日常工作中的積累和沉淀。
Cookie 是一小段文本信息,伴隨著用戶請求和頁面在 Web 服務(wù)器和瀏覽器之間傳遞。Cookie 包含每次用戶訪問站點時 Web 應(yīng)用程序都可以讀取的信息。
例如,如果在用戶請求站點中的頁面時應(yīng)用程序發(fā)送給該用戶的不僅僅是一個頁面,還有一個包含日期和時間的 Cookie,用戶的瀏覽器在獲得頁面的同時還獲得了該 Cookie,并將它存儲在用戶硬盤上的某個文件夾中。
以后,如果該用戶再次請求您站點中的頁面,當(dāng)該用戶輸入 URL 時,瀏覽器便會在本地硬盤上查找與該 URL 關(guān)聯(lián)的 Cookie。如果該 Cookie 存在,瀏覽器便將該 Cookie 與頁請求一起發(fā)送到您的站點。然后,應(yīng)用程序便可以確定該用戶上次訪問站點的日期和時間。您可以使用這些信息向用戶顯示一條消息,也可以檢查到期日期。
Cookie 與網(wǎng)站關(guān)聯(lián),而不是與特定的頁面關(guān)聯(lián)。因此,無論用戶請求站點中的哪一個頁面,瀏覽器和服務(wù)器都將交換 Cookie 信息。用戶訪問不同站點時,各個站點都可能會向用戶的瀏覽器發(fā)送一個 Cookie;瀏覽器會分別存儲所有 Cookie。
Cookie 幫助網(wǎng)站存儲有關(guān)訪問者的信息。一般來說,Cookie 是一種保持 Web 應(yīng)用程序連續(xù)性(即執(zhí)行狀態(tài)管理)的方法。除短暫的實際交換信息的時間外,瀏覽器和 Web 服務(wù)器間都是斷開連接的。對于用戶向 Web 服務(wù)器發(fā)出的每個請求,Web 服務(wù)器都會單獨處理。但是在很多情況下,Web 服務(wù)器在用戶請求頁時識別出用戶會十分有用。例如,購物站點上的 Web 服務(wù)器跟蹤每位購物者,這樣站點就可以管理購物車和其他的用戶特定信息。因此,Cookie 可以作為一種名片,提供相關(guān)的標(biāo)識信息幫助應(yīng)用程序確定如何繼續(xù)執(zhí)行。
使用 Cookie 能夠達到多種目的,所有這些目的都是為了幫助網(wǎng)站記住用戶。例如,一個實施民意測驗的站點可以簡單地將 Cookie 作為一個 Boolean 值,用它來指示用戶的瀏覽器是否已參與了投票,這樣用戶便無法進行第二次投票。要求用戶登錄的站點則可以通過 Cookie 來記錄用戶已經(jīng)登錄,這樣用戶就不必每次都輸入憑據(jù)。
Cookies保存在用戶的本地機器上,不同的瀏覽器存儲在不同的文件夾中,并且按照域名分別保存。即網(wǎng)站之間的Cookies不會彼此覆蓋。
IE瀏覽器的用戶可以通過在本地的文檔中找到Cookies的txt文件, 不同操作系統(tǒng)的位置不同,windows server 2003/xp都保存在:
C:\Documents and Settings\Administrator\Cookies 文件夾下。
其中名稱txt按照域名保存,比如localhost域下的cookies為:
administrator@localhost[1].txt 或者 administrator@localhost[2].txt
其中后面的[1]和[2]是隨著每次保存交替變化的。
Cookies的信息是在Web服務(wù)器和瀏覽器之間傳遞的。保存在Http請求中。
在請求一個頁面的Http頭中,會將屬于此頁面的本地Cookies信息加在Http頭中,注意下面加粗的部分:
GET /Cookies/Test.aspx HTTP/1.1 Host: localhost:1335 User-Agent: Mozilla/5.0 (Windows; U; Windows NT 5.2; zh-CN; rv:1.9.1.1) Gecko/20090715 Firefox/3.5.1 GTB5 (.NET CLR 3.5.30729) Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8 Accept-Language: zh-cn,zh;q=0.5 Accept-Encoding: gzip,deflate Accept-Charset: GB2312,utf-8;q=0.7,*;q=0.7 Keep-Alive: 300 Connection: keep-alive Cookie: My.Common.TestCookieInfo=Pkid=999&TestValue=aaabbbcccdddeee
如果頁面要求寫入Cookies信息,則返回的Http如下,注意加粗的部分:
HTTP/1.x 200 OK
Server: ASP.NET Development Server/9.0.0.0
Date: Thu, 06 Aug 2009 03:40:59 GMT
X-AspNet-Version: 2.0.50727
Set-Cookie: My.Common.TestCookieInfo=Pkid=999&TestValue=aaabbbcccdddeee; expires=Fri, 07-Aug-2009 03:40:59 GMT; path=/
Cache-Control: private
Content-Type: text/html; charset=utf-8
Content-Length: 558
Connection: Close
IE用戶可以直接查看Cookies的txt文件。
比如:C:\Documents and Settings\Administrator\Cookies\administrator@localhost[1].txt
FF下使用Web Developer插件可以很方便的查看、刪除和修改Cookies:
插件截圖:
查看頁面Cookies:
大多數(shù)瀏覽器支持最大為 4096 字節(jié)的 Cookie。
瀏覽器還限制站點可以在用戶計算機上存儲的 Cookie 的數(shù)量。大多數(shù)瀏覽器只允許每個站點存儲 20 個 Cookie;注意這里的20個是指主鍵值,也就是20條Cookies記錄,但是每個Cookies記錄還可以包含若干子鍵,下面會詳細解釋。如果試圖存儲更多 Cookie,則最舊的 Cookie 便會被丟棄。有些瀏覽器還會對它們將接受的來自所有站點的 Cookie 總數(shù)作出絕對限制,通常為 300 個。
Cookies可以包含一個主鍵, 主鍵再包含子鍵。比如asp.net中獲取Cookies的格式是:
Request.Cookies[key][subkey].ToString();
其中的key就是主鍵,subkey就是主鍵關(guān)聯(lián)的子鍵。
My.Common.TestCookieInfo Pkid=999&TestValue=aaabbbcccdddeee localhost/ 1536 3059603968 30021392 2348960464 30021191 *
其中的Pkid=999&TestValue=aaabbbcccdddeee 是Cookies的值,由于使用了subkey=subvalue的格式, 所以此Cookies是包含子鍵的。
在Javascript中給的Cookie是一個字符串,通過document.cookies獲取。字符格式如下:
My.Common.SubKey=Pkid=999&TestValue=aaabbbcccdddeee; SingleKey=SingleKeyValue
上面的字符串包含了兩個Cookies,一個是不包含子鍵的SingleKey, 一個是包含pkid和TextValue兩個子鍵的My.Common.SubKey,兩個Cookie通過“;”分割。
和所有的服務(wù)器端語言一樣,Asp.Net中使用集合類保存Cookies集合:
public sealed class HttpCookieCollection : NameObjectCollectionBase {...}
通過HttpResquest和HttpResponse對象的Cookies屬性,可以獲取和寫入當(dāng)前頁面的Cookies。
Cookies的值中可以保存除了“;”以外的標(biāo)點符號。但是不能保存漢字。保存漢字會出現(xiàn)亂碼。
所以對于Cookies中的內(nèi)容要進行統(tǒng)一的編碼和解碼。為了在瀏覽器端和服務(wù)器端都能夠進行解碼和編碼, 所以要統(tǒng)一使用UTF編碼格式。
主要是因為javascript中只能使用UTF編碼格式。
Cookies的Path屬性表示當(dāng)前的Cookies可以作用在網(wǎng)站的那個路徑下。
比如下面的兩個同名的Cookies:
允許存在兩個同名但是Path不同的Cookies。
無論是服務(wù)器端還是客戶端,在獲取時優(yōu)先獲取本頁路徑下面的Cookies。
也就是說如果在、/chapter10/路徑下面的頁面, 獲取testKey這個Cookies的值,則只能獲取到testValue222222這個值。
如果保存Cookies時未設(shè)置過期時間, 則Cookies的過期時間為“當(dāng)前瀏覽器進程有效”,即和Session一樣關(guān)閉瀏覽器后則消失。在asp.net中還可以通過設(shè)置HttpCookie對象的過期時間為DateTime.MinValue來指定此Cookies為跟隨瀏覽器生效。(這句話來之不易啊,在腦袋等人的幫助下才查到的。)
如果設(shè)置了過期時間并且大于當(dāng)前時間,則會保存Cookies值。
如果設(shè)置了過期時間但是小于等于當(dāng)前時間,則清除Cookies值。
有時我們會忽略Cookies與Session的關(guān)系。但是兩者是密不可分的。
Session的唯一標(biāo)示:SessionID是通常保存在Cookies中的(也可以保存在URL中)。對于Asp.Net而言,SessionID保存在鍵值為“ASP.NET_SessionId”的Cookies中,如圖:
因為Cookies的存儲數(shù)量是有限制的,所以我們的系統(tǒng)在保存Cookies的時候一定要注意防止沖掉這一個關(guān)鍵的Cookies。在下文介紹的最佳實踐-以強對象方式保存Cookies的代碼中特意對這個Cookies做了處理。
注意,在客戶端使用javascript腳本無法獲取“ASP.NET_SessionId”的Cookies, 因為此Cookies在服務(wù)器端設(shè)置了HttpOnly屬性為true。
ASP.Net中HttpCookie對象的HttpOnly 屬性 指定一個Cookie 是否可通過客戶端腳本訪問。不能通過客戶端腳本訪問為 true;否則為 false。默認值為 false。此屬性并不能完全阻止客戶端在本地獲取cookies,但是可以增加通過腳本直接獲取的難度。
Microsoft Internet Explorer 版本 6 Service Pack 1 和更高版本支持 Cookie 屬性 HttpOnly。
在設(shè)置Cookies的屬性時,有一個選項Secure用來控制Cookie的加密特性。
如果通過 SSL 連接 (HTTPS) 傳輸 Cookie,則為 true;否則為 false。默認為 false。
如果我們保存一個Cookies并設(shè)置加密,那么在非HTTPS的頁面中,無論是使用javascript還是服務(wù)器端都無法獲得此Cookies。但是在本地依然可以看到此Cookies的存在。
如果Ajax請求訪問一個服務(wù)器頁面,此服務(wù)器頁面是可以向用戶瀏覽器寫入Cookies和Session的。
在了解了Cookies的相關(guān)知識后,下面提出最佳的事件方法。其中包括客戶端和服務(wù)器端兩部分。
通常,我們使用Request和Response對象來直接操作Cookies:
寫入Cookies:
Response.Cookies["k1"].Value = "k1Value"; Response.Cookies["k2"]["k2-1"] = "k2-1Value"; Response.Cookies.Add(new HttpCookie("k3", "k3Value"));
讀取Cookies:
Request["k1"] ; Request.Cookies["k1"].Value ; Request.Cookies["k2"]["k2-1"]; Request.Cookies.Get(0).Value;
注意Request["k1"]這個大家熟悉的獲取get和post參數(shù)的方法,同時還能夠獲取Cookies的值!
另外上面語句中的有些是必須通過Value屬性訪問的,有些則不需要。
下面提供一個可以以對象方式整體保存Cookies的工具類。并且只占用一條Cookies,所有的屬性都存在子鍵上。
/// <summary> /// Cookies基類。將需要保存Cookies的數(shù)據(jù)類此類派生,可以將強類型對象在Cookies中的保存和讀取。 /// </summary> /// <remarks> /// 2009.8.6 ziqiu.zhang created /// </remarks> /// <example> /// 假設(shè)MyCookiesInfo是從 從Cookies中獲取對象: /// <code> /// CookieInfo item = new CookieInfo(); //new以后已經(jīng)從Cookies中構(gòu)造了對象。 /// </code> /// 將對象保存在Cookies中: /// <code> /// CookieInfo item = new CookieInfo(); /// item.value = "test value"; /// item.SetCookies("1"); //Cookies有效期為1天 /// </code> /// </example> [System.Serializable] public class CookieInfo { #region ==================== Constructed Method ==================== /// <summary> /// 構(gòu)造函數(shù) /// </summary> public CookieInfo() { } #endregion #region ==================== Public Method ==================== /// <summary> /// 得到當(dāng)前Cookies的過期時間 /// </summary> /// <returns>過期時間</returns> public DateTime GetExpiresTime() { string cookieName = GetType().ToString(); if (HttpContext.Current.Request.Cookies[cookieName] != null) { return HttpContext.Current.Request.Cookies[cookieName].Expires; } return DateTime.MinValue; } /// <summary> /// 保存Cookies,過期時間為瀏覽器關(guān)閉則失效。 /// </summary> /// <param name="expiresTime">Cookies過期事件</param> /// <returns>是否保存成功</returns> public bool Save() { return this.Save(DateTime.MinValue); } /// <summary> /// 保存Cookies,需要指定過期時間。 /// </summary> /// <param name="expiresTime">Cookies過期事件</param> /// <returns>是否保存成功</returns> public bool Save(DateTime expiresTime) { string CookieName = GetType().ToString(); HttpCookie SessionCookie = null; //對 SessionId 進行備份. if (HttpContext.Current.Request.Cookies["ASP.NET_SessionId"] != null) { string SesssionId = HttpContext.Current.Request.Cookies["ASP.NET_SessionId"].Value.ToString(); SessionCookie = new HttpCookie("ASP.NET_SessionId"); SessionCookie.Value = SesssionId; } //設(shè)定cookie 過期時間. DateTime dtExpiry = expiresTime; HttpContext.Current.Response.Cookies[CookieName].Expires = dtExpiry; //設(shè)定cookie 域名. string domain = string.Empty; if (HttpContext.Current.Request.Params["HTTP_HOST"] != null) { //domain = "www.elong.com"; domain = HttpContext.Current.Request.Params["HTTP_HOST"].ToString(); } //如果是www.elong.com或多級域名,需要轉(zhuǎn)化為elong.com if (domain.IndexOf(".") > -1) { string[] temp = domain.Split('.'); if (temp.Length >= 3) { domain = temp[temp.Length - 2].Trim() + "." + temp[temp.Length - 1].Trim(); } HttpContext.Current.Response.Cookies[CookieName].Domain = domain; } //把類的屬性, 寫入Cookie. PropertyInfo[] Propertys = GetType().GetProperties(); foreach (PropertyInfo pi in Propertys) { object oj = pi.GetValue(this, null); Type type = pi.PropertyType; string valueStr = string.Empty; if (oj != null && oj.ToString() != string.Empty) { if (type == Type.GetType("System.DateTime")) { valueStr = ((DateTime)oj).ToString("yyyy/MM/dd HH:mm:ss", System.Globalization.DateTimeFormatInfo.InvariantInfo); } else { valueStr = oj.ToString(); } HttpContext.Current.Response.Cookies[CookieName][pi.Name] = HttpUtility.UrlEncode(valueStr); } } //如果cookie總數(shù)超過20 個, 重寫ASP.NET_SessionId, 以防Session 丟失. if (HttpContext.Current.Request.Cookies.Count > 20 && SessionCookie != null) { if (SessionCookie.Value != string.Empty) { HttpContext.Current.Response.Cookies.Remove("ASP.NET_SessionId"); HttpContext.Current.Response.Cookies.Add(SessionCookie); } } return true; } /// <summary> /// 找回Cookie值 /// </summary> public void Load() { string cookieValue = string.Empty; string CookieName = GetType().ToString(); //通過遍歷屬性, 從cookie 中找回值, 回寫到屬性. PropertyInfo[] Propertys = GetType().GetProperties(); foreach (PropertyInfo pi in Propertys) { try { cookieValue = HttpUtility.UrlDecode(HttpContext.Current.Request.Cookies[CookieName][pi.Name].ToString()); } catch { cookieValue = string.Empty; } if (pi.CanWrite && cookieValue != null && cookieValue != string.Empty) { try { object obb = cookieValue; Type type = pi.PropertyType; obb = Convert.ChangeType(obb, type); pi.SetValue(this, obb, null); } catch { } } } } #endregion }
首先說明如何使用此類。
為想要保存在Cookies中的類建立模型,并且繼承自CookieInfo即可。比如下面建立了MyCookieInfo類,其中包含屬性pkid,TestValue和TestDateTime:
/// <summary> /// 保存Cookies的數(shù)據(jù)對象 /// </summary> [System.Serializable] public class MyCookieInfo : CookieInfo { private int m_Pkid = 0; public int Pkid { get { return m_Pkid ; } set { m_Pkid = value ; } } private string m_TestValue = ""; public string TestValue { get { return m_TestValue; } set { m_TestValue = value; } } private DateTime m_TestDateTime = DateTime.Now; public DateTime TestDateTime { get { return m_TestDateTime; } set { m_TestDateTime = value; } } }
接下來就可以使用對象的Save和Load方法保存和讀取Cookies:
MyCookieInfo testCookies = new MyCookieInfo(); testCookies.Pkid = 1; testCookies.TestValue = "中文測試"; testCookies.Save();
MyCookieInfo testCookies = new MyCookieInfo(); testCookies.Load(); this.lblMsg.Text = "Pkid:" + testCookies.Pkid.ToString(); this.lblMsg.Text += ",TestValue:" + testCookies.TestValue.ToString(); this.lblMsg.Text += ",TestDateTime:" + testCookies.TestDateTime.ToString("yyyy/MM/dd HH:mm:ss", System.Globalization.DateTimeFormatInfo.InvariantInfo);
現(xiàn)在我們已經(jīng)可以將一個強類型的對象讀取和保存Cookies了。
在客戶端我們同樣需要操作Cookies。
下面是封裝了的專門用于操作Cookies的jQuery工具函數(shù)。如果還有人不知道jQuery是什么,請參考我的“從零開始學(xué)習(xí)jQuery”系列教程:
http://www.cnblogs.com/zhangziqiu/archive/2009/04/30/jQuery-Learn-1.html
當(dāng)然此工具函數(shù)稍加修改,就可以變成標(biāo)準的Javascript函數(shù)。
下載地址:http://files.cnblogs.com/zhangziqiu/jquery.extend-lastest.js
工具函數(shù)說明:
方法簽名: jQuery.cookie(name, subName, value, options)
方法說明:讀取、寫入、刪除Cookies
方法參數(shù):
名稱 | 說明 | 舉例 |
name | cookies的主鍵值 |
讀取主鍵:
$.cookie("singleKey")
寫入cookies,值為字符串: $.cookie("singleKey", "", "singleKey-value", { expires: 1, path: "/", secure: false }) |
subName | 子鍵名稱。在寫入時請傳遞空或者null |
讀取子鍵:$.cookie("multiKey", "subName1") 寫入cookies,值為對象: var subNameObj = { subName1: "aaa", subName2: "bbb", subName3: "ccc" }; $.cookie("multiKey", "", subNameObj, { expires: 1, path: "/", secure: false }); |
value |
Cookies值,可以是字符串或者對象。 如果是對象,則將對象的每個屬性保存在Cookies子鍵。 |
參見上面實例。 |
options |
參數(shù): expires:可以是數(shù)字或者Data類型的對象。 如果傳入數(shù)字表示幾天后過期。 path:路徑,默認為域名根目錄(“/”)。 secure:是否啟用加密,默認為否。 |
指定過期時間: var myDate = new Date(); myDate.setFullYear(2009, 10, 10); $.cookie("singleKey", "", "singleKey-value", { expires: myDate, secure: false }) 1天后過期: var time = Date(); $.cookie("singleKey", "", "singleKey-value", { expires: 1, path: "/", secure: false }) |
很久沒有發(fā)表文章了,作為博客園改版后我的第一篇文章, 希望對大家的工作能有所幫助。 歡迎拍磚!
如對本文有疑問,請?zhí)峤坏浇涣髡搲?,廣大熱心網(wǎng)友會為你解答?。?點擊進入論壇