php 新session_PHP重写session机制

众所周知,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中的一个例子中的部分代码。大家可以点此查看完整代码。希望本文对大家有帮助。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值