就php来讲,语言自己支持的session是以文件的方式保存到磁盘文件中,保存在指定的文件夹中,保存的路径能够在配置文件中设置或者在程序中使用函数session_save_path()进行设置,可是这么作有弊端,php
第一就是保存到文件系统中,效率低,只要有用到session就会从好多个文件中查找指定的sessionid,效率很低。html
第二就是当用到多台服务器的时候可能会出现,session丢失问题(实际上是保存在了其余服务器上)。mysql
固然了,保存在缓存中能够解决上面的问题,若是使用php自己的session函数,可使用session_set_save_handler()函数很方便的对session的处理过程进行从新控制。若是不用php的session系列函数,能够本身编写个相似的session函数,也是能够的,我如今作的这个项目就是这样,会根据用户的mid、登陆时间进行求hash做为sessionId,每次请求的时候都必须加上sessionId才算合法(第一次登陆的时候是不须要的,这个时候会建立sessionId,返回给客户端),这么作也很方便、简洁高效的。固然了,我这篇文章主要说的是在php自身的SESSION中”作作手脚”。redis
php将缓存保存到redis中,可使用配置文件,对session的处理和保存作修改,固然了,在程序中使用ini_set()函数去修改也能够,这个很方便测试,我这里就使用这种方式,固然了,要是生产环境仍是建议使用配置文件。sql
<?php ini_set("session.save_handler", "redis"); ini_set("session.save_path", "tcp://localhost:6379"); session_start(); header("Content-type:text/html;charset=utf-8"); if(isset($_SESSION['view'])){ $_SESSION['view'] = $_SESSION['view'] + 1; }else{ $_SESSION['view'] = 1; } echo "【view】{$_SESSION['view']}";
这里设置session.save_handler方式为redis,session.save_path为redis的地址和端口,设置以后刷新,再回头查看redis,会发现redis中的生成了sessionId,sessionId和浏览器请求的是同样的,数据库
是否是很方便呢,只须要改下配置文件就能够实现redis中保存session,可是我这里要说的是经过程序的方式来处理session保存到redis或者db,下面一块儿来看看。数组
这里能够先看看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); //保存数据的时候调用
destroy($sessionId); //销毁会话的时候(session_destroy()或者session_regenerate_id())会调用
gc($lifeTime); //垃圾清理函数,清理掉过时做废的数据
主要就是实现这几个方法,根据不一样的存储驱动能够本身设置不一样的具体方法,我实现了mysql数据库和redis这两种保存session的驱动,若是有须要的话能够本身去扩展,扩展很方便很容易。
下面是个人redis的实现(db和redis差很少,redis代码少,贴出来):
我使用了接口的方式,这样扩展起来更方便,那天想用memcached了,直接添加就好了
<?php include_once __DIR__."/interfaceSession.php"; /** * 以db的方式存储session */ class redisSession implements interfaceSession{ /** * 保存session的数据库表的信息 */ private $_options = array( 'handler' => null, //数据库链接句柄 'host' => null, 'port' => null, 'lifeTime' => null, ); /** * 构造函数 * @param $options 设置信息数组 */ public function __construct($options=array()){ if(!class_exists("redis", false)){ die("必须安装redis扩展"); } if(!isset($options['lifeTime']) || $options['lifeTime'] <= 0){ $options['lifeTime'] = ini_get('session.gc_maxlifetime'); } $this->_options = array_merge($this->_options, $options); } /** * 开始使用该驱动的session */ public function begin(){ if($this->_options['host'] === null || $this->_options['port'] === null || $this->_options['lifeTime'] === null ){ return false; } //设置session处理函数 session_set_save_handler( array($this, 'open'), array($this, 'close'), array($this, 'read'), array($this, 'write'), array($this, 'destroy'), array($this, 'gc') ); } /** * 自动开始回话或者session_start()开始回话后第一个调用的函数 * 相似于构造函数的做用 * @param $savePath 默认的保存路径 * @param $sessionName 默认的参数名,PHPSESSID */ public function open($savePath, $sessionName){ if(is_resource($this->_options['handler'])) return true; //链接redis $redisHandle = new Redis(); $redisHandle->connect($this->_options['host'], $this->_options['port']); if(!$redisHandle){ return false; } $this->_options['handler'] = $redisHandle; $this->gc(null); return true; } /** * 相似于析构函数,在write以后调用或者session_write_close()函数以后调用 */ public function close(){ return $this->_options['handler']->close(); } /** * 读取session信息 * @param $sessionId 经过该Id惟一肯定对应的session数据 * @return session信息/空串 */ public function read($sessionId){ return $this->_options['handler']->get($sessionId); } /** * 写入或者修改session数据 * @param $sessionId 要写入数据的session对应的id * @param $sessionData 要写入的数据,已经序列化过了 */ public function write($sessionId, $sessionData){ return $this->_options['handler']->setex($sessionId, $this->_options['lifeTime'], $sessionData); } /** * 主动销毁session会话 * @param $sessionId 要销毁的会话的惟一id */ public function destroy($sessionId){ return $this->_options['handler']->delete($sessionId) >= 1 ? true : false; } /** * 清理绘画中的过时数据 * @param 有效期 */ public function gc($lifeTime){ //获取全部sessionid,让过时的释放掉 $this->_options['handler']->keys("*"); return true; } }
看看简单工厂模式
class session { /** * 驱动程序句柄保存 */ private static $_handler = null; /** * 建立session驱动程序 */ public static function getSession($type, $options){ //单例 if(isset($handler)){ return self::$_handler; } switch ($type) { case 'db': //数据库驱动session类型 include_once __DIR__."/driver/dbSession.php"; $handler = new dbSession($options); break; case 'redis': //redis驱动session类型 include_once __DIR__."/driver/redisSession.php"; $handler = new redisSession($options); break; default: return false; break; } return self::$_handler = $handler; } }
调用也很简单,
session::getSession('redis',array( 'host' => "localhost", 'port' => "6379", ))->begin(); session_start();
数据库版本的也同样很简单就能够配置,须要的话能够在这里下载完整版和demo
本文版权归做者iforever(luluyrt@163.com)全部,未经做者本人赞成禁止任何形式的转载,转载文章以后必须在文章页面明显位置给出做者和原文链接,不然保留追究法律责任的权利。