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
// 引入初始化文件
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.

原文地址:http://blog.51cto.com/13447608/2295961

时间: 2024-08-02 07:22:09

php负载中使用redis实现session会话保持的相关文章

分布式中使用Redis实现Session共享(二)

上一篇介绍了一些redis的安装及使用步骤,本篇开始将介绍redis的实际应用场景,先从最常见的session开始,刚好也重新学习一遍session的实现原理.在阅读之前假设你已经会使用nginx+iis实现负载均衡搭建负载均衡站点了,这里我们会搭建两个站点来验证redis实现的session是否能共享. 阅读目录 Session实现原理 session共享实现方案 问题拓展 总结 回到顶部 Session实现原理 session和cookie是我们做web开发中常用到的两个对象,它们之间会不会

node.js中结合redis实现session修改完善(2)

代码的设计比较乱,整理了jade模板之后,决定链接登录注册功能,但发现很多的实现都太凑合了,先修改了httpParam的get, post, cookie 方法.现在决定修改Session,因为session太无厘头了. 1. 结合的比较混乱 2. session和redis操作在一起,没有独立出去 3. session中存放的数据不明朗 4. session中设置了过期,其实没有必要,因为在redis中已经设置了过期时间 5. 页面切换的时候,没有对session过期时间进行更新 下面即之前的

Nginx+Tomcat负载平衡,Redis管理session存储

使用Nginx作为Tomcat的负载平衡器,Tomcat的会话Session数据存储在Redis,能够实现0当机的7x24运营效果.因为将会话存储在Redis中,因此Nginx就不必配置成stick粘粘某个Tomcat方式,这样才能真正实现后台多个Tomcat负载平衡,用户请求能够发往任何一个tomcat主机,当我们需要部署新应用代码时,只要停止任何一台tomcat,所有当前在线用户都会导向到运行中的tomcat实例,因为会话数据被序列化到Redis,在线用户不会受到影响,一旦停掉的tomcat

分布式中使用Redis实现Session共享(转)

上一篇介绍了如何使用nginx+iis部署一个简单的分布式系统,文章结尾留下了几个问题,其中一个是"如何解决多站点下Session共享".这篇文章将会介绍如何使用Redis,下一篇在此基础上实现Session. 这里特别说明一下,其实没有必要使用Redis来解决Session共享.Asp.net提供了StateServer模式来共享Session,这里重复造轮子的目的1:熟悉Redis的基本知识和使用 2.学习和巩固Session的实现原理. 3.学习Redis应用场景 阅读目录 Re

分布式中使用Redis实现Session共享(一)

上一篇介绍了如何使用nginx+iis部署一个简单的分布式系统,文章结尾留下了几个问题,其中一个是"如何解决多站点下Session共享".这篇文章将会介绍如何使用Redis,下一篇在此基础上实现Session. 这里特别说明一下,其实没有必要使用Redis来解决Session共享.Asp.net提供了StateServer模式来共享Session,这里重复造轮子的目的1:熟悉Redis的基本知识和使用 2.学习和巩固Session的实现原理. 3.学习Redis应用场景 阅读目录 Re

node.js中结合redis实现session

这里的session机制很简单,用户登录之后,给用户生成一个session,包含用户的唯一编号ID,过期时间expires,以及用户的用户的sessionID. 创建了session之后,将session存放到redis数据库中,现在也只是测试了可以存放数据,并没有进行redis的进一步操作,今后将对redis进行集群操作,虽然小,但是一定要五脏俱全,会写两个类,一个专门用来写,针对主库,然后另一个专门用来读,针对从库. 这需要一个key值与redis从库的连接地址相关联,因为当存在多个key值

Nginx反向代理,负载均衡,redis session共享,keepalived高可用

相关知识自行搜索,直接上干货... 使用的资源: nginx主服务器一台,nginx备服务器一台,使用keepalived进行宕机切换. tomcat服务器两台,由nginx进行反向代理和负载均衡,此处可搭建服务器集群. redis服务器一台,用于session的分离共享. nginx主服务器:192.168.50.133 nginx备服务器:192.168.50.135 tomcat项目服务器1:192.168.50.137 tomcat项目服务器2:192.168.50.139 redis服

Redis慢查询日志 PHP中使用redis – 安装扩展模块

Redis慢查询日志编辑配置文件/etc/redis.conf针对慢查询日志,可以设置两个参数,一个是执行时长,单位是微秒,另一个是慢查询日志的长度.当一个新的命令被写入日志时,最老的一条会从命令日志队列中被移除. slowlog-log-slower-than 1000 //单位ms,表示慢于1000ms则记录日志slowlog-max-len 128 //定义日志长度,表示最多存128条slowlog get //列出所有的慢查询日志slowlog get 2 //只列出2条slowlog

负载均衡session会话保持方法

负载均衡时,为了保证同一用户session会被分配到同一台服务器上,可以使用以下方法:1.使用cookie将用户的session存入cookie里,当用户分配到不同的服务器时,先判断服务器是否存在该用户的session,如果没有就先把cookie里面的sessoin存入该服务器,实现session会话保持.缺点是存入cookie有安全隐患. 2.使用缓存利用memcache,redis等缓存分布式的特点,可以将所有服务器产生的session存入同一台服务器的缓存中,实现session共享.这样安