本文實例講述了thinkPHP多域名情況下使用memcache方式共享session數據的實現方法。分享給大家供大家參考,具體如下:
一、問題起源
稍大一些的網站,通常都會有好幾個服務器,每個服務器運行著不同功能的模塊,使用不同的二級域名,而一個整體性強的網站,用戶系統是統一的,即一套用戶名、密碼在整個網站的各個模塊中都是可以登錄使用的。各個服務器共享用戶數據是比較容易實現的,只需要在後端放個數據庫服務器,各個服務器通過統一接口對用戶數據進行訪問即可。但還存在一個問題,就是用戶在這個服務器登錄之後,進入另一個服務器的別的模塊時,仍然需要重新登錄,這就是一次登錄,全部通行的問題,映射到技術上,其實就是各個服務器之間如何實現共享 SESSION 數據的問題。
二、PHP SESSION 的工作原理
在解決問題之前,先來了解一下 PHP SESSION 的工作原理。在客戶端(如浏覽器)登錄網站時,被訪問的 PHP 頁面可以使用session_start() 打開 SESSION,這樣就會產生客戶端的唯一標識 SESSION ID(此 ID 可通過函數session_id() 獲取/設置)。SESSION ID 可以通過兩種方式保留在客戶端,使得請求不同的頁面時,PHP程序可以獲知客戶端的 SESSION ID;一種是將 SESSION ID 自動加入到 GET 的 URL 中,或者 POST的表單中,默認情況下,變量名為 PHPSESSID;另一種是通過 COOKIE,將 SESSION ID 保存在 COOKIE中,默認情況下,這個 COOKIE 的名字為 PHPSESSID。這裡我們主要以 COOKIE方式進行說明,因為應用比較廣泛。
那麼 SESSION 的數據保存在哪裡呢?當然是在服務器端,但不是保存在內存中,而是保存在文件或數據庫中。默認情況下,php.ini中設置的 SESSION 保存方式是 files(session.save_handler = files),即使用讀寫文件的方式保存SESSION 數據,而 SESSION 文件保存的目錄由 session.save_path 指定,文件名以 sess_為前綴,後跟 SESSIONID,如:sess_c72665af28a8b14c0fe11afe3b59b51b。文件中的數據即是序列化之後的 SESSION數據了。如果訪問量大,可能產生的 SESSION 文件會比較多,這時可以設置分級目錄進行 SESSION文件的保存,效率會提高很多,設置方法為:session.save_path=”N;/save_path”,N為分級的級數,save_path 為開始目錄。當寫入 SESSION 數據的時候,PHP 會獲取到客戶端的SESSION_ID,然後根據這個 SESSION ID 到指定的 SESSION 文件保存目錄中找到相應的 SESSION文件,不存在則創建之,最後將數據序列化之後寫入文件。讀取 SESSION數據是也是類似的操作流程,對讀出來的數據需要進行解序列化,生成相應的 SESSION 變量。
三、多服務器共享 SESSION 的主要障礙及解決辦法
通過了解 SESSION 的工作原理,我們可以發現,在默認情況下,各個服務器會各自分別對同一個客戶端產生 SESSIONID,如對於同一個用戶浏覽器,A 服務器產生的 SESSION ID 是30de1e9de3192ba6ce2992d27a1b6a0a,而 B 服務器生成的則是c72665af28a8b14c0fe11afe3b59b51b。另外,PHP 的 SESSION數據都是分別保存在本服務器的文件系統中。
確定了問題所在之後,就可以著手進行解決了。想要共享 SESSION 數據,那就必須實現兩個目標:一個是各個服務器對同一個客戶端產生的SESSION ID 必須相同,並且可通過同一個 COOKIE 進行傳遞,也就是說各個服務器必須可以讀取同一個名為 PHPSESSID的 COOKIE;另一個是 SESSION 數據的存儲方式/位置必須保證各個服務器都能夠訪問到。簡單地說就是多服務器共享客戶端的SESSION ID,同時還必須共享服務器端的 SESSION 數據。
第一個目標的實現其實很簡單,只需要對 COOKIE 的域(domain)進行特殊地設置即可,默認情況下,COOKIE的域是當前服務器的域名/IP 地址,而域不同的話,各個服務器所設置的 COOKIE 是不能相互訪問的,如 www.aaa.com的服務器是不能讀寫 www.bbb.com 服務器設置的 COOKIE的。這裡我們所說的同一網站的服務器有其特殊性,那就是他們同屬於同一個一級域,如:tieba.xiaoyuan.com 和www.xiaoyuan.com 都屬於域 .xiaoyuan.com,那麼我們就可以設置 COOKIE 的域為.xiaoyuan.com,這樣 tieba.xiaoyuan.com、www.xiaoyuan.com 等等都可以訪問此COOKIE。PHP 代碼中的設置方法如下:
<?php ini_set('session.cookie_domain', '.xiaoyuan.com'); ?>
這樣各個服務器共享同一客戶端 SESSION ID 的目的就達到了。
第二個目標的實現可以使用文件共享方式,有2種方式可以解決,一是用數據庫存session,還有就是試用memcache。這裡用MEMCACHE來解決.
我用的是thinkphp框架,已經支持memcache方式存取session.架好memcache服務器後,只需要在配置文件裡面設置好memcache的IP和端口,然後指定COOKIE_DOMAIN參數就可以了,然後就可以按正常操作session的方式進行操作,這時已經可以多域名共享session了
更多關於thinkPHP相關內容感興趣的讀者可查看本站專題:《ThinkPHP入門教程》、《thinkPHP模板操作技巧總結》、《ThinkPHP常用方法總結》、《smarty模板入門基礎教程》及《PHP模板技術總結》。
希望本文所述對大家基於ThinkPHP框架的PHP程序設計有所幫助。