最近在改session的时候惊讶的发现了一个很奇怪的问题,一个session刚刚建立,第一次访问的时候是没有问题的,第二次访问就直接不存在了!其由来是我们现在需要做一个session管理平台跨多语言的,所以做了一个远端的共享session,但是为了让原有的使用方式能正常使用,并保证共享session异常能够短时间的正常使用!
所以我们做了如下的操作,我们做了一个自己的session组件(我们团队中大神级写的)
/**
* 设置session变量
* @see CHttpSession::add() *
* @param string $key session变量名称
* @param mixed $value session变量值
*/
public function add( $key, $value ) {
$this->listener->add( $key, $value );
parent::add( $key, $value );
}
public function get( $key, $defaultValue=null ) {
$result = $this->listener->get( $key, $this->order );
if ( is_null( $result ) ) {
return parent::get( $key, $defaultValue );
}
return $result;
}
这里是主要的写入和读取的方式,很明显在写入的时候我们进行了2次写入,所以我们可以使用yii->session-get()以及$_SESSION['KEY'],一些不需要共享的可以不使用get方法,同事为了很好的利用监听的功能(下一章简要介绍一下),重写了父类的readSession,writeSession等方法 ,直接$this->_cache->set去调用memcache缓存,因为在CHttpsession中opan方法中@session_set_save_handler(array($this,'openSession'),array($this,'closeSession'),array($this,'readSession'),array($this,'writeSession'),array($this,'destroySession'),array($this,'gcSession'));
是这样进行描述的,也就是当子类重写此类方法的时候,这些系统级方法将会被替换,这样parent::add的实质也是用到了自己改写的writeSession了,那么在回顾一下writeSession
public function writeSession($id,$data) {
return $this->_cache->set($this->calculateKey($id),$data,$this->getTimeout()); //getTimeout(父类CHttpsession中的)是内部调用PHP.INI的设置时间,默认是永久
}
那么我们查看一下在Cmemcache中的setvalue方法
protected function setValue($key,$value,$expire)
{
if($expire>0)
$expire+=time();
else
$expire=0;
return $this->useMemcached ? $this->_cache->set($key,$value,$expire) : $this->_cache->set($key,$value,0,$expire);
}
问题出现了, if($expire>0)$expire+=time();,这得多少啊,然而Memcache的最长缓存时间3600*24*30,也就是30天,你去看看time()+,那岂不是只要$expire不为0那就永远碉堡了,这里明显出现了时间使用定义的分歧了,但是为什么会显示一次,然后第二次就会抛异常,session被置空我就不明白为什么了!