YII 内部问题 session和memcache小问题

来源:互联网 发布:淘宝主营类目如何更改 编辑:程序博客网 时间:2024/04/29 23:55

        最近在改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被置空我就不明白为什么了!

原创粉丝点击