為什麼要把SESSION保存在快取
就php來說,語言本身支援的session是以檔案的方式保存到磁碟檔案中,保存在指定的資料夾中,保存的路徑可以在設定檔中設定或在程式中使用函數session_save_path()進行設置,但是這麼做有弊端,
第一就是儲存到檔案系統中,效率低,只要有用到session就會從好多個檔案中找出指定的sessionid,效率很低。
第二就是當用到多台伺服器的時候可能會出現,session遺失問題(其實是保存在了其他伺服器上)。
當然了,保存在快取中可以解決上面的問題,如果使用php本身的session函數,可以使用session_set_save_handler()函數很方便的對session的處理過程進行重新控制。如果不用php的session系列函數,可以自己寫個類似的session函數,也是可以的,我現在做的這個專案就是這樣,會根據使用者的mid、登入時間進行求hash作為sessionId,每次要求的時候都必須加上sessionId才算合法(第一次登入的時候是不需要的,這個時候會建立sessionId,回傳給客戶端),這麼做也很方便、簡潔有效率的。當然了,我這篇文章主要說的是在php自身的SESSION中」做做手腳」。
SESSION保存在快取中
php將快取保存到redis中,可以使用設定文件,對session的處理和保存做修改,當然了,在程式中使用ini_set()函數去修改也可以,這個很方便測試,我在這裡就使用這種方式,當然了,要是生產環境還是建議使用設定檔。
<?<span>php </span><span>ini_set</span>("session.save_handler", "redis"<span>); </span><span>ini_set</span>("session.save_path", "tcp://localhost:6379"<span>); </span><span>session_start</span><span>(); </span><span>header</span>("Content-type:text/html;charset=utf-8"<span>); </span><span>if</span>(<span>isset</span>(<span>$_SESSION</span>['view'<span>])){ </span><span>$_SESSION</span>['view'] = <span>$_SESSION</span>['view'] + 1<span>; }</span><span>else</span><span>{ </span><span>$_SESSION</span>['view'] = 1<span>; } </span><span>echo</span> "【view】{<span>$_SESSION</span>['view']}";
這裡設定session.save_handler方式為redis,session.save_path為redis的地址和端口,設置之後刷新,再回頭查看redis,會發現redis中的生成了sessionId,sessionId和瀏覽器請求的是一樣的,
是不是很方便呢,只需要改下設定檔就可以實現redis中保存session,但是我這裡要說的是透過程式的方式來處理session保存到redis或db,下面一起來看看。
透過php提供的接口,自己改寫session的處理函數
這裡可以先看看php的這個函數session_set_save_handler,php5.4及之後可以直接實作SessionHandlerInterface接口,程式碼會更簡潔。重寫的時候主要有以下幾個方法
open(string $savePath, string $sessionName); //open類似於建構函數,開始會話的時候會調用,例如使用session_start()函數之後
close(); //類似於類別的析構函數,在write函數調用之後調用,session_write_close()之後也會執行
read(string $sessionId); //讀取session的時候呼叫
write(string $sessionId, string $data); //儲存資料的時候呼叫
destory($sessionId); //銷毀會話的時候(session_destory()或session_regenerate_id())會呼叫
gc($lifeTime); //垃圾清理函數,清理掉過期作廢的資料
主要就是實現這幾個方法,根據不同的存儲驅動可以自己設置不同的具體方法,我實現了mysql數據庫和redis這兩種保存session的驅動,如果有需要的話可以自己去擴展,擴展很方便很容易。
以下是我的redis的實作(db和redis差不多,redis程式碼少,貼出來):
我使用了介面的方式,這樣擴充起來比較方便,那天想用memcached了,直接加入就行了
<?<span>php </span><span>include_once</span> __DIR__."/interfaceSession.php"<span>; </span><span>/*</span><span>* * 以db的方式存储session </span><span>*/</span> <span>class</span> redisSession <span>implements</span><span> interfaceSession{ </span><span>/*</span><span>* * 保存session的数据库表的信息 </span><span>*/</span> <span>private</span> <span>$_options</span> = <span>array</span><span>( </span>'handler' => <span>null</span>, <span>//</span><span>数据库连接句柄</span> 'host' => <span>null</span>, 'port' => <span>null</span>, 'lifeTime' => <span>null</span>,<span> ); </span><span>/*</span><span>* * 构造函数 * @param $options 设置信息数组 </span><span>*/</span> <span>public</span> <span>function</span> __construct(<span>$options</span>=<span>array</span><span>()){ </span><span>if</span>(!<span>class_exists</span>("redis", <span>false</span><span>)){ </span><span>die</span>("必须安装redis扩展"<span>); } </span><span>if</span>(!<span>isset</span>(<span>$options</span>['lifeTime']) || <span>$options</span>['lifeTime'] <= 0<span>){ </span><span>$options</span>['lifeTime'] = <span>ini_get</span>('session.gc_maxlifetime'<span>); } </span><span>$this</span>->_options = <span>array_merge</span>(<span>$this</span>->_options, <span>$options</span><span>); } </span><span>/*</span><span>* * 开始使用该驱动的session </span><span>*/</span> <span>public</span> <span>function</span><span> begin(){ </span><span>if</span>(<span>$this</span>->_options['host'] === <span>null</span> || <span>$this</span>->_options['port'] === <span>null</span> || <span>$this</span>->_options['lifeTime'] === <span>null</span><span> ){ </span><span>return</span> <span>false</span><span>; } </span><span>//</span><span>设置session处理函数</span> <span>session_set_save_handler</span><span>( </span><span>array</span>(<span>$this</span>, 'open'), <span>array</span>(<span>$this</span>, 'close'), <span>array</span>(<span>$this</span>, 'read'), <span>array</span>(<span>$this</span>, 'write'), <span>array</span>(<span>$this</span>, 'destory'), <span>array</span>(<span>$this</span>, 'gc'<span>) ); } </span><span>/*</span><span>* * 自动开始回话或者session_start()开始回话后第一个调用的函数 * 类似于构造函数的作用 * @param $savePath 默认的保存路径 * @param $sessionName 默认的参数名,PHPSESSID </span><span>*/</span> <span>public</span> <span>function</span> open(<span>$savePath</span>, <span>$sessionName</span><span>){ </span><span>if</span>(<span>is_resource</span>(<span>$this</span>->_options['handler'])) <span>return</span> <span>true</span><span>; </span><span>//</span><span>连接redis</span> <span>$redisHandle</span> = <span>new</span><span> Redis(); </span><span>$redisHandle</span>->connect(<span>$this</span>->_options['host'], <span>$this</span>->_options['port'<span>]); </span><span>if</span>(!<span>$redisHandle</span><span>){ </span><span>return</span> <span>false</span><span>; } </span><span>$this</span>->_options['handler'] = <span>$redisHandle</span><span>; </span><span>$this</span>->gc(<span>null</span><span>); </span><span>return</span> <span>true</span><span>; } </span><span>/*</span><span>* * 类似于析构函数,在write之后调用或者session_write_close()函数之后调用 </span><span>*/</span> <span>public</span> <span>function</span><span> close(){ </span><span>return</span> <span>$this</span>->_options['handler']-><span>close(); } </span><span>/*</span><span>* * 读取session信息 * @param $sessionId 通过该Id唯一确定对应的session数据 * @return session信息/空串 </span><span>*/</span> <span>public</span> <span>function</span> read(<span>$sessionId</span><span>){ </span><span>return</span> <span>$this</span>->_options['handler']->get(<span>$sessionId</span><span>); } </span><span>/*</span><span>* * 写入或者修改session数据 * @param $sessionId 要写入数据的session对应的id * @param $sessionData 要写入的数据,已经序列化过了 </span><span>*/</span> <span>public</span> <span>function</span> write(<span>$sessionId</span>, <span>$sessionData</span><span>){ </span><span>return</span> <span>$this</span>->_options['handler']->setex(<span>$sessionId</span>, <span>$this</span>->_options['lifeTime'], <span>$sessionData</span><span>); } </span><span>/*</span><span>* * 主动销毁session会话 * @param $sessionId 要销毁的会话的唯一id </span><span>*/</span> <span>public</span> <span>function</span> destory(<span>$sessionId</span><span>){ </span><span>return</span> <span>$this</span>->_options['handler']->delete(<span>$sessionId</span>) >= 1 ? <span>true</span> : <span>false</span><span>; } </span><span>/*</span><span>* * 清理绘画中的过期数据 * @param 有效期 </span><span>*/</span> <span>public</span> <span>function</span> gc(<span>$lifeTime</span><span>){ </span><span>//</span><span>获取所有sessionid,让过期的释放掉</span> <span>$this</span>->_options['handler']->keys("*"<span>); </span><span>return</span> <span>true</span><span>; } }</span>
看看簡單工廠模式
<span>class</span><span> session { </span><span>/*</span><span>* * 驱动程序句柄保存 </span><span>*/</span> <span>private</span> <span>static</span> <span>$_handler</span> = <span>null</span><span>; </span><span>/*</span><span>* * 创建session驱动程序 </span><span>*/</span> <span>public</span> <span>static</span> <span>function</span> getSession(<span>$type</span>, <span>$options</span><span>){ </span><span>//</span><span>单例</span> <span>if</span>(<span>isset</span>(<span>$handler</span><span>)){ </span><span>return</span> self::<span>$_handler</span><span>; } </span><span>switch</span> (<span>$type</span><span>) { </span><span>case</span> 'db': <span>//</span><span>数据库驱动session类型</span> <span>include_once</span> __DIR__."/driver/dbSession.php"<span>; </span><span>$handler</span> = <span>new</span> dbSession(<span>$options</span><span>); </span><span>break</span><span>; </span><span>case</span> 'redis': <span>//</span><span>redis驱动session类型</span> <span>include_once</span> __DIR__."/driver/redisSession.php"<span>; </span><span>$handler</span> = <span>new</span> redisSession(<span>$options</span><span>); </span><span>break</span><span>; </span><span>default</span>: <span>return</span> <span>false</span><span>; </span><span>break</span><span>; } </span><span>return</span> self::<span>$_handler</span> = <span>$handler</span><span>; } }</span>
呼叫也很簡單,
session::getSession('redis',<span>array</span><span>( </span>'host' => "localhost", 'port' => "6379",<span> ))</span>-><span>begin(); </span><span>session_start</span>();
資料庫版本的也一樣很簡單就可以配置,需要的話可以在這裡下載完整版和demo
本文版權歸作者iforever(luluyrt@163.com)所有,未經作者本人同意禁止任何形式的轉載,轉載文章之後必須在文章頁面明顯位置給出作者和原文連接,否則保留追究法律責任的權利。
以上就介紹了session放入快取(redis)、DB,包含了方面的內容,希望對PHP教學有興趣的朋友有幫助。