新聞中心
今天遇到了這個(gè)問題,于是研究了一下。要解決這個(gè)問題,首先就要明白一些Session的機(jī)理。Session在服務(wù)器是以散列表形式存在的,我們都知道Session是會(huì)話級(jí)的,每個(gè)用戶訪問都會(huì)生成一個(gè)Session。那么服務(wù)器是怎么區(qū)分不同用戶的Session?又是怎么將不同用戶的Session與不同的用戶綁定的呢?下面我們來研究一下,以下純屬我個(gè)人的理解,如有錯(cuò)誤請(qǐng)指證。

公司主營(yíng)業(yè)務(wù):成都網(wǎng)站制作、成都網(wǎng)站設(shè)計(jì)、外貿(mào)營(yíng)銷網(wǎng)站建設(shè)、移動(dòng)網(wǎng)站開發(fā)等業(yè)務(wù)。幫助企業(yè)客戶真正實(shí)現(xiàn)互聯(lián)網(wǎng)宣傳,提高企業(yè)的競(jìng)爭(zhēng)能力。成都創(chuàng)新互聯(lián)是一支青春激揚(yáng)、勤奮敬業(yè)、活力青春激揚(yáng)、勤奮敬業(yè)、活力澎湃、和諧高效的團(tuán)隊(duì)。公司秉承以“開放、自由、嚴(yán)謹(jǐn)、自律”為核心的企業(yè)文化,感謝他們對(duì)我們的高要求,感謝他們從不同領(lǐng)域給我們帶來的挑戰(zhàn),讓我們激情的團(tuán)隊(duì)有機(jī)會(huì)用頭腦與智慧不斷的給客戶帶來驚喜。成都創(chuàng)新互聯(lián)推出沾化免費(fèi)做網(wǎng)站回饋大家。
Session在服務(wù)器端是以散列表的形式存在的,區(qū)分每一個(gè)Session是通過SessionID來實(shí)現(xiàn)的,所以可以說這個(gè)SessionID是一個(gè)Key是一個(gè)全局唯一的值。我們可以通過ASP.NET來打印出SessionID,如下代碼:
- protected void Page_Load(object sender, EventArgs e)
- {
- Response.Write(Session.SessionID.ToString());
- }
這樣我們就得到了這樣的值:0julmoedn0kz3gyfnr1vksv0,有點(diǎn)像是GUID,就算不是算法也都是類似的,主要就是為了保證全局唯一性。這樣就達(dá)到了區(qū)分不同用戶的Session的目的。接下來還有第二個(gè)問題,那就是SessionID有了,但是它又是怎么和相應(yīng)的訪問者(用戶)綁定的呢?比如說用戶A訪問維護(hù)了自己的SessionID,用戶B訪問也維護(hù)了自己的SessionID。我們都知道web是基于http無鏈接的,他們又是怎么做到的呢?沒錯(cuò),答案就是在客戶端存儲(chǔ)了自己的SessionID。瀏覽器存儲(chǔ)SessionID有兩種方式,一種就是利用Cookies;還有一種就是利用url參數(shù)(這種我們不常用,很不友好)。
話題說到Cookies上來了,怎么的?沒想到Session和Cookies還有這樣的關(guān)系吧?(很多人知道,別BS我)沒錯(cuò),當(dāng)我們請(qǐng)求一個(gè)URL時(shí)候,服務(wù)器會(huì)生成一個(gè)全局的SessionID,并且把這個(gè)值以Cookies的形式保存在客戶端也就是瀏覽器(這里暫不討論url方式)。這樣當(dāng)用戶再去請(qǐng)求的時(shí)候,在http頭把這個(gè)SessionID的Cookie發(fā)到服務(wù)器端,服務(wù)器就去找這個(gè)SessionID,如果找到了。就證明這個(gè)用戶的狀態(tài)是存在的。
知道了這個(gè)原理,我們的問題也就有眉頭了,即然是用Cookies來保存SessionID,那么我們就可以在Cooikes上做手腳了。我們都知道Cooikes記錄方式是以域(例如:http://www.local.com/)為區(qū)分的,這也是各種瀏覽器規(guī)定的。如果不這么做,安全性就會(huì)有問題。我們要做的就是讓指定Cookies的父域方式,不指定具體指域,這樣Cookies就可以跨子域了。Cookies可以像這樣指定域:
- protected void Page_Load(object sender, EventArgs e)
- {
- Response.Cookies["MyCook"].Domain = ".local.com";
- }
這樣,我們所有的二級(jí)域全部是認(rèn)這一個(gè)主域的,比如a.local.com;b.local.com;user.local.com等等。有了這個(gè)認(rèn)識(shí),我想大家心里也有數(shù)了,該怎么怎么做,但是現(xiàn)在問題是用來生成SessionID的方法是ASP.NET自動(dòng)實(shí)現(xiàn)的,我們又怎么去干涉它呢?這是這樣做的,不主動(dòng)干涉它,但是我可以操作它的Cookies啊。接下來我們就研究ASP.NET存SessionID的Cooike的名字是什么。經(jīng)過網(wǎng)上很容易就查找到了,名字是:ASP.NET_SessionId,這個(gè)就是SessionId的Cookies名字。我們可以在Session_Start中這樣寫:
- protected void Session_Start(object sender, EventArgs e)
- {
- Response.Cookies["ASP.NET_SessionId"].Value = Session.SessionID.ToString();
- Response.Cookies["ASP.NET_SessionId"].Domain = ".local.com";
- }
代碼的意思是每次會(huì)話開始的時(shí)候,我都把ASP.NET_SessionId這個(gè)Cookie重寫成我們已有的SessionID,并且把這個(gè)Cookie的domain指定為父域,比如:.local.com,這樣就可以實(shí)現(xiàn)跨子域的Session共享了。怎么樣很簡(jiǎn)單吧?
我們還有一個(gè)外題問題,就是客戶端保存的問題解決了,但是服務(wù)器端的Session怎么辦?一般情況下我們不同的子域做的是指向不同的服務(wù)器的,比如user.local.com 專門一臺(tái)服務(wù)器,yellow.local.com專門一臺(tái)服務(wù)器。這時(shí)它們別說是進(jìn)程了,連物理上都不是一個(gè)了。Session怎么共享?這時(shí)就用到另一個(gè)方法了,我們默認(rèn)的Session是存儲(chǔ)在asp.net進(jìn)程中的,這樣沒法互相訪問,如下面所示:
我們可以修改為State Server方式,這是一個(gè)單獨(dú)的服務(wù)可以用來存儲(chǔ)ASP.NET Session的,它支持分布式遠(yuǎn)程主機(jī)的,這樣我們可以用一臺(tái)服務(wù)器來提供Session服務(wù),如下所示:
- ="tcpip=127.0.0.1:42424" timeout="30" />
這樣,就完全實(shí)現(xiàn)了不同子域的Session共享了。
前面說到Url保存SessionId的方式,由于不常用,給大家演示一下,如下配置就可以了:
- "tcpip=127.0.0.1:42424" timeout="30" cookieless="true" />
cookieless屬性指定是否用cookie來保存SessionId,我們運(yùn)行一下得到下面的樣子:
http://localhost:3380/(S(dqxcs455n4u2vg55ia51fvqg))/default.aspx
分享標(biāo)題:詳解ASP.NET在不同的子域中共享Session
當(dāng)前網(wǎng)址:http://fisionsoft.com.cn/article/cojscip.html


咨詢
建站咨詢
