一、session的本質
http協議是無狀態的,即你連續訪問某個網頁100次和訪問1次對server來說是沒有差別對待的,由於它記不住你。
那么,在一些場合,確實須要server記住當前用戶怎么辦?比方用戶登錄郵箱后,接下來要收郵件、寫郵件,總不能每次操作都讓用戶輸入username和password吧,為了解決問題,session的方案就被提了出來,其實它並非什么新技術,並且也不能脫離http協議以及不論什么現有的web技術。
原理非常easy,如果你訪問網頁時就像逛澡堂,第一次進去你是沒有鑰匙的,這個時候你交了錢服務台就分配一把鑰匙給你,你走到哪里都要帶上,由於這是你身份的唯一標識,接下來你用這把鑰匙能夠去打開一個專有的儲物櫃存儲你的衣物,游完泳,你再用鑰匙去打開櫃子拿出衣物,最后離開游泳池時,把鑰匙歸還,你的這次游泳的過程就是一次session,或者叫做會話,在這個樣例中,鑰匙就是session的key,而儲物櫃能夠理解為存儲用戶會話信息的介質。
那么在web server中怎樣實現session呢?想必看了上面的樣例你會非常easy理解,主要是解決兩個問題,一個是鑰匙的問題,一個是存儲用戶信息的問題。對於第一個問題,即什么東西能夠讓你每次請求都會自己主動帶到server呢?假設你比較了解http協議,那么答案一目了然,就是cookie,假設你想為用戶建立一次會話,能夠在用戶授權成功時給他一個cookie,叫做會話id,它當然是唯一的,比方php就會為建立會話的用戶默認set一個名為phpsessid,值看起來為一個隨機字符串的cookie,假設下次發現用戶帶了這個cookie,server就知道,哎呀,剛剛這位顧客來了。
剩下的是解決第二個問題,即怎樣存儲用戶的信息,server知道會話id為abc的用戶來了,那abc想存儲自己的私人信息,比方購物車信息,怎樣處理?這個時候能夠用內存、也能夠用文件,也能夠用數據庫了,但有個要求是,數據須要用用戶的會話id就可以取到,比方php就默認會把會話id為abc的用戶會話數據存儲到/tmp/phpsess_abc的文件中面,每次讀取都要反序列化程序能夠理解的數據,寫的時候又須要序列化為持久的數據格式。
較好理解的描寫敘述:
session被用於表示一個持續的連接狀態,在站點訪問中一般指代client瀏覽器的進程從開啟到結束的過程。session事實上就是站點分析的訪問(visits)度量,表示一個訪問的過程。
session的常見實現形式是會話cookie(session cookie),即未設置過期時間的cookie,這個cookie的默認生命周期為瀏覽器會話期間,僅僅要關閉瀏覽器窗體,cookie就消失了。實現機制是當用戶發起一個請求的時候,server會檢查該請求中是否包括sessionid,假設未包括,則系統會創造一個名為JSESSIONID的輸出 cookie返回給瀏覽器(僅僅放入內存,並不存在硬盤中),並將其以HashTable的形式寫到server的內存里面;當已經包括sessionid是,服務端會檢查找到與該session相匹配的信息,假設存在則直接使用該sessionid,若不存在則又一次生成新的 session。這里須要注意的是session始終是有服務端創建的,並不是瀏覽器自己生成的。 可是瀏覽器的cookie被禁止后session就須要用get方法的URL重寫的機制或使用POST方法提交隱藏表單的形式來實現。
二、怎樣實現session的共享?
首先我們應該明確,為什么要實現共享,假設你的站點是存放在一個機器上,那么是不存在這個問題的,由於會話數據就在這台機器,可是假設你使用了負載均衡把請求分發到不同的機器呢?這個時候會話id在client是沒有問題的,可是假設用戶的兩次請求到了兩台不同的機器,而它的session數據可能存在當中一台機器,這個時候就會出現取不到session數據的情況,於是session的共享就成了一個問題。
1.各種web框架早已考慮到這個問題,比方asp.net,是支持通過配置文件改動session的存儲介質為sql server的,全部機器的會話數據都從同一個數據庫讀,就不會存在不一致的問題;
2.以cookie加密的方式保存在client.長處是減輕server端的壓力,缺點是受到cookie的限制大小,可能占用一定帶寬,由於每次請求會在頭部附帶一定大小的cookie信息,另外這樣的方式在用戶禁止使用cookie的情況下無效.
3.server間同步。定時同步各個server的session信息,此方法可能有一定延時,用戶體驗也不是非常好。
4.php支持把會話數據存儲到某台memcacheserver,你也能夠手工把session文件存放的文件夾改為nfs網絡文件系統,從而實現文件的跨機器共享。
另一個簡單的辦法能夠用於會話信息不會頻繁變更的情況,在機器a設置用戶會話的時候,把會話數據post到機器b的一個cgi,機器b的cgi把會話數據存下來,這樣機器a和b都會有同一份session數據的拷貝。
