众所周知,session在web应用中占有举足轻重的地位。而且,在很多情况下我们需要改变session的存储位置。当然了,改变session存储的位置可以在php.ini文件中直接修改。但是,这需要我们对服务器有足够的权限。可是事实却是在很多时候我们并没有权限去操作php.ini文件的权限。这时需要我们通过PHP提供的session_set_save_handler()函数来重写session。

针对这一情况,PHP 5 > 5.4和php 7 支持SessionHandlerInterface 接口。

SessionHandlerInterface {
/* 方法 */
abstract public bool close ( void )
abstract public bool destroy ( string $session_id )
abstract public bool gc ( int $maxlifetime )
abstract public bool open ( string $save_path , string $name )
abstract public string read ( string $session_id )
abstract public bool write ( string $session_id , string $session_data )
}
我们需要做的就是实现这个接口中的所有的方法。然后通过session_set_save_handler()函数来使方法生效。
注:本文中的例子是将session存到redis中。对于PHP如何操redis,大家可以参考《PHP操作Redis的两种方式》。
下面我们来分别介绍这些函数的实现方法。
open函数
abstract public bool SessionHandlerInterface ::open ( string $save_path , string $name ){}
重新初始化现有的session,抑或是创建一个session。该函数在session_start()函数执行的时候被调用。
$save_path 这个参数对应的就是php.ini中的session.save_path选项。这个选项设置的值就是$save_path的值。默认情况下,php.ini中session.save_path这个选项是被注释的,所以$save_path的值为空。举个例子:session.save_path设置为/tmp ,则$save_path的值为/tmp。
$name 这个参数对应的就是php.ini中的 session.name选项。默认情况下session.name设置为PHPSESSID。 所以说$name参数值为PHPSESSID。
我在实现这个函数的时候没有做其他的处理(因为我想将session存到redis中),只是连接了redis数据库。
public function open($save_path, $name){
/*
* 首先连接服务器
*/
$this->parseConnect();
return true;
}
close函数
abstract public bool SessionHandlerInterface :: close(){}
关闭当前的session。该函数在当关闭session的时候被自动触发,或者在程序中调用session_write_close()函数是触发close()函数。
在实现该函数时没有做什么特殊的处理
public function close(){
return true;
}
read函数
abstract public bool SessionHandlerInterface :: read($session_id){}
读取session数据。当调用session_start()函数的时候会触发read()函数。当然该函数的触发是在open之后的。
$session_id 该参数就是对应的由客户端传过来的sessionId。所有的操作都需要根据这个sessionId来进行。
public function read($session_id){
/*
* 根据sessionId 构造键名
*/
$key = $this->prefix.':'.$session_id;
//读取当前sessionid下的data数据
$res = $this->handle->hGet($key,'data');
//读取完成以后 更新时间,说明已经操作过session
$this->handle->hSet($key,'last_time',time());
return $res;
}
write函数
abstract public bool SessionHandlerInterface :: write($session_id , $session_data){}
该函数是将session的数据写到相应的位置去。当操作$_SESSION来序列化数据的时候该函数被触发。
$session_id 这个参数就是我们上面所说的sessionId。
$session_data 是我们要存储的数据。这里需要说明一下,我们通过$_SESSION来设置数据的时候,第一次我们设置了$_SESSION[‘login’] = ‘ok’,这时$session_data的值为login|s:2:ok。write会把login|s:2:ok写入到redis中。然后我们不退出session,再次设置$_SESSION[‘name’] = ‘onmpw’。这时$session_data的值不只是name|s:5:onmpw,而是login|s:2:ok;name|s:5:onmpw。因为在write之前会先触发read来读取redis中的数据。读取到数据以后将这些数据连同通过$_SESSION[‘name’] = ‘onmpw’得到的值一块儿作为$session_data参数的值来进行更新。
public function write($session_id, $session_data){
/*
* 根据sessionId 构造键名
*/
$key = $this->prefix.':'.$session_id;
//查看该键内容是否存在
if(!$this->handle->exists($key)){
/*
* 不存在则插入新的内容
* 插入最后更新时间
*/
$this->handle->hset($key,'last_time',time());
}else{
/*
* 存在,则更新该键值
*/
$this->handle->hMset($key,array('last_time'=>time(),'data'=>$session_data));
}
return true;
}
destroy函数
abstract public bool SessionHandlerInterface ::destroy($session_id){}
当函数session_destroy()调用的时候触发该函数。这时我们可以在该函数中将$session_id对应的数据销毁掉
public function destroy($session_id){
/*
* 根据sessionId 构造键名
*/
$key = $this->prefix.':'.$session_id;
$this->handle->hDel($key,'data');
}
gc函数
abstract public bool SessionHandlerInterface ::gc($maxlifetime){}

清除垃圾session,也就是清除过期的session。该函数是基于php.ini中的配置选项:session.gc_divisor, session.gc_probability 和 session.gc_lifetime所设置的值的。

$maxlifetime 参数的值就是 session.gc_lifetime选项所设置的值。

这个函数是否被触发要取决于session.gc_divisor和session.gc_probability这两个选项。该函数被触发的概率为 session.gc_probability/session.gc_divisor。如果probability设置为1,divisor设置为100。那么gc函数被触发的概率就是1%。也就是说在100个请求中可能会在某一个请求过程中触发这个函数。从这里我们可以知道,如果客户端一直没有请求,那这个函数就永远不会被触发。即使有些session信息没被操作的时间已经超过了session.gc_lifetime所设置的时间。

那什么是过期的session呢?这么来说吧,假如session.gc_lifetime设置的值为30(默认单位为s 秒),一条session信息,从最后一次被操作的时间开始计时,如果在30秒内没有再被操作,那这条session就被定为垃圾信息了。当gc函数被触发的时候这条信息就被清除掉了。如果说,你在30秒内又对这条session信息进行了操作——即使是在29s的时候,那这条session信息会在你最近操作的这一时刻开始再重新计时30秒。

public function gc($maxlifetime){
/*
* 取出所有的 带有指定前缀的键
*/
$keys = $this->handle->keys($this->prefix.'*');
$now =time(); //取得现在的时间
foreach($keys as $key){
//取得当前key的最后更新时间
$last_time = $this->handle->hGet($key,'last_time');
/*
* 查看当前时间和最后的更新时间的时间差是否超过最大生命周期
*/
if(($now - $last_time) > $maxlifetime){
//超过了最大生命周期时间 则删除该key
$this->handle->del($key);
}
}
}

上面就是我们在重写session机制中会用到的几个函数。上面的部分代码是我写的将session保存到redis中的一个例子中的部分代码。