专栏首页首富手记php负载中使用redis实现session会话保持

php负载中使用redis实现session会话保持

首先要明确session和cookie的区别。浏览器端存的是cookie每次浏览器发请求到服务端是http 报文头是会自动加上你的cookie信息的。服务端拿着用户的cookie作为key去存储里找对应的value(session). 同一域名下的网站的cookie都是一样的。所以无论几台服务器,无论请求分配到哪一台服务器上同一用户的cookie是不变的。也就是说cookie对应的session也是唯一的。 所以,这里只要保证多台业务服务器访问同一个redis服务器(或集群)就行了。

修改php会话缓存机制改成Redis即可,这里有三种方式: 1,修改php的配置文件 修改php.ini文件

session.save_handler = redis
session.save_path = "tcp://172.16.1.51:6379"
//session.save_path = "tcp://172.16.1.51:6379?auth=123123"如果redis配置的密码需要写成这种方式,填写redis的密码
session.auto_start = 1

注释php-fpm.d/www.conf里面的两条内容

;php_value[session.save_handler] = files
;php_value[session.save_path]    = /var/lib/php/session

更改完成之后一定要重启php-fpm服务

2,更改对应的代码文件 直接在对应的代码中加上下面两句话

ini_set("session.save_handler","redis");
ini_set("session.save_path","tcp://172.16.1.51:6379");
//如果redis设置了密码需要改下面一句为
//ini_set("session.save_path","tcp://172.16.1.51:6379?auth=123123");

测试

<?php
//ini_set("session.save_handler", "redis");
//ini_set("session.save_path", "tcp://172.16.1.51:6379?auth=123123");

session_start();

//存入session
$_SESSION['class'] = array('name' => 'toefl', 'num' => 8);

//连接redis
$redis = new redis();
$redis->connect('127.0.0.1', 6379);

//检查session_id
echo 'session_id:' . session_id() . '<br/>';

//redis存入的session(redis用session_id作为key,以string的形式存储)
echo 'redis_session:' . $redis->get('PHPREDIS_SESSION:' . session_id()) . '<br/>';

//php获取session值
echo 'php_session:' . json_encode($_SESSION['class']);

3,自定义会话机制(目前不懂) 使用 session_set_save_handle 方法自定义会话机制,网上发现了一个封装非常好的类,我们可以直接使用这个类来实现我们的共享session操作。

<?php
class redisSession{
    /**
     * 保存session的数据库表的信息
     */
    private $_options = array(
        'handler' => null, //数据库连接句柄
        'host' => null,
        'port' => null,
        'lifeTime' => null,
        'prefix'   => 'PHPREDIS_SESSION:'
    );

    /**
     * 构造函数
     * @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, 'destory'),
            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){
        $sessionId = $this->_options['prefix'].$sessionId; 
        return $this->_options['handler']->get($sessionId);
    }

    /**
     * 写入或者修改session数据
     * @param $sessionId 要写入数据的session对应的id
     * @param $sessionData 要写入的数据,已经序列化过了
     */
    public function write($sessionId, $sessionData){
        $sessionId = $this->_options['prefix'].$sessionId; 
        return $this->_options['handler']->setex($sessionId, $this->_options['lifeTime'], $sessionData);
    }

    /**
     * 主动销毁session会话
     * @param $sessionId 要销毁的会话的唯一id
     */
    public function destory($sessionId){
        $sessionId = $this->_options['prefix'].$sessionId; 
//        $array = $this->print_stack_trace();
//        log::write($array);
        return $this->_options['handler']->delete($sessionId) >= 1 ? true : false;
    }

    /**
     * 清理绘画中的过期数据
     * @param 有效期
     */
    public function gc($lifeTime){
        //获取所有sessionid,让过期的释放掉
        //$this->_options['handler']->keys("*");
        return true;
    }
    //打印堆栈信息
    public function print_stack_trace()
    {
        $array = debug_backtrace ();
        //截取用户信息
        $var = $this->read(session_id());
        $s = strpos($var, "index_dk_user|");
        $e = strpos($var, "}authId|");
        $user = substr($var,$s+14,$e-13);
        $user = unserialize($user);
        //print_r($array);//信息很齐全
        unset ( $array [0] );
        if(!empty($user)){
          $traceInfo = $user['id'].'|'.$user['user_name'].'|'.$user['user_phone'].'|'.$user['presona_name'].'++++++++++++++++\n';
        }else{
          $traceInfo = '++++++++++++++++\n';
        }
        $time = date ( "y-m-d H:i:m" );
        foreach ( $array as $t ) {
            $traceInfo .= '[' . $time . '] ' . $t ['file'] . ' (' . $t ['line'] . ') ';
            $traceInfo .= $t ['class'] . $t ['type'] . $t ['function'] . '(';
            $traceInfo .= implode ( ', ', $t ['args'] );
            $traceInfo .= ")\n";
        }
        $traceInfo .= '++++++++++++++++';
        return $traceInfo;
            }
    }

在你的项目入口处调用上边的类:上边的方法等于是重写了session写入文件的方法,将数据写入到了Redis中。
初始化文件 init.php
<?php
require_once("redisSession.php");
$handler = new redisSession(array(
'host' => "172.16.1.51",
'port' => "6379"
));
$handler->begin();
// 这也是必须的,打开session,必须在session_set_save_handler后面执行
session_start();
测试 test.php
```php
<?php
// 引入初始化文件
include("init.php");
$_SESSION['isex'] = "Hello";  
$_SESSION['sex']  = "Corwien";

// 打印文件
print_r($_SESSION);
// ( [sex] => Corwien [isex] => Hello )

在Redis客户端使用命令查看我们的这条数据是否存在:

172.16.1.51:6379> keys *
 1) "first_key"
 2) "mylist"
 3) "language"
 4) "mytest"
 5) "pragmmer"
 6) "good"
 7) "PHPREDIS_SESSION:29a111bcs120sv48ibmmjqdag4"
 8) "user:1"
 9) "counter:__rand_int__"
10) "key:__rand_int__"
11) "tutorial-list"
12) "id:1"
13) "name"
127.0.0.1:6379> get PHPREDIS_SESSION:29a111bcs120sv48ibmmjqdag4
"sex|s:7:\"Corwien\";isex|s:5:\"Hello\";"
127.0.0.1:6379>

我们可以看到,我们的数据被保存在了Redis端了,键为: PHPREDIS_SESSION:29a111bcs120sv48ibmmjqdag4.

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • prometheus 告警

    告警能力在Prometheus的架构中被划分成两个独立的部分。如下所示,通过在Prometheus中定义AlertRule(告警规则),Prometheus会周...

    张琳兮
  • Nginx+php更改了fastcgi_pass后面的地址php不能正常请求

    1.1.1 通过yum安装Nginx和php,更改了Nginx里面fastcgi_pass后的地址php不能正常请求 1.1.1.1 问题还原: Nginx...

    张琳兮
  • docker 仓库里面python好多tag都代表什么意思?我们该如何选择

    今天让我同事帮忙构建一个基于python代码的docker包,然后他问我使用那个底层镜像,我说你直接去docker hub上找一个,他打开之后问我这么多我该使用...

    张琳兮
  • Django 2.1.7 Session基本操作,解决 'WSGIRequest' object has no attribute 'session' 问题

    上一篇Django 2.1.7 状态保持 - Cookie介绍了Django中关于cookie的基本使用,本篇章继续来看看session的操作。

    Devops海洋的渔夫
  • JSP中session的11个常用方法——收藏备查

    session中的属性在当前session中是共享的 session表示一个请求的javax.servlet.http.HttpSession对象。指的是客户端...

    用户1289394
  • 通过EmbeddedServletContainerCustomizer接口调优Tomcat

    通过在application.properties设置对应的key-value对,可以配置Spring Boot应用程序的很多特性,例如POST、SSL、MyS...

    阿杜
  • Shiro实战(五) - 会话管理

    Shiro提供了完整的企业级会话管理功能,不依赖于底层容器(如web容器Tomcat),不管JavaSE还是JavaEE环境都可以使用,提供了会话管理、会话事件...

    JavaEdge
  • 贝壳产品面经

    参加了贝壳的提前批,笔试没过,当时觉得非常遗憾,然后前几天贝壳来哈尔滨宣讲,可以线下笔试,就抱着一定要进一次面试的决心去了,笔试后的第二天下午收到了面试通知,当...

    牛客网
  • Hey,Siri,帮我把服务器A的X目录凌晨五点拷贝到B服务器上

    人无法从海量的语料中学习到规律,但是语料经过数学化后,经历深度网络,网络的的节点通过某种群体行为能够记录下这种规律,从而在新的数据到来后,能够用这种隐藏的规律进...

    用户2936994
  • AlphaGo首次与人类并肩作战,双打配合产分歧

    大数据文摘

扫码关注云+社区

领取腾讯云代金券