session放入缓存(redis)、DB

为什么要把SESSION保存在缓存

就php来说,语言本身支持的session是以文件的方式保存到磁盘文件中,保存在指定的文件夹中,保存的路径可以在配置文件中设置或者在程序中使用函数session_save_path()进行设置,但是这么做有弊端,

第一就是保存到文件系统中,效率低,只要有用到session就会从好多个文件中查找指定的sessionid,效率很低。

第二就是当用到多台服务器的时候可能会出现,session丢失问题(其实是保存在了其他服务器上)。

当然了,保存在缓存中可以解决上面的问题,如果使用php本身的session函数,可以使用session_set_save_handler()函数很方便的对session的处理过程进行重新控制。如果不用php的session系列函数,可以自己编写个类似的session函数,也是可以的,我现在做的这个项目就是这样,会根据用户的mid、登录时间进行求hash作为sessionId,每次请求的时候都必须加上sessionId才算合法(第一次登录的时候是不需要的,这个时候会创建sessionId,返回给客户端),这么做也很方便、简洁高效的。当然了,我这篇文章主要说的是在php自身的SESSION中”做做手脚”。

SESSION保存在缓存中

php将缓存保存到redis中,可以使用配置文件,对session的处理和保存做修改,当然了,在程序中使用ini_set()函数去修改也可以,这个很方便测试,我这里就使用这种方式,当然了,要是生产环境还是建议使用配置文件。

<?php
ini_set("session.save_handler", "redis");
ini_set("session.save_path", "tcp://localhost:6379");
session_start();
header("Content-type:text/html;charset=utf-8");
if(isset($_SESSION[‘view‘])){
    $_SESSION[‘view‘] = $_SESSION[‘view‘] + 1;
}else{
    $_SESSION[‘view‘] = 1;
}
echo "【view】{$_SESSION[‘view‘]}";

这里设置session.save_handler方式为redis,session.save_path为redis的地址和端口,设置之后刷新,再回头查看redis,会发现redis中的生成了sessionId,sessionId和浏览器请求的是一样的,

是不是很方便呢,只需要改下配置文件就可以实现redis中保存session,但是我这里要说的是通过程序的方式来处理session保存到redis或者db,下面一起来看看。

通过php提供的接口,自己改写session的处理函数

这里可以先看看php的这个函数 session_set_save_handler ,php5.4及之后可以直接实现 SessionHandlerInterface 接口,代码会更加简洁。重写的时候主要有下面几个方法

open(string $savePath, string $sessionName); //open类似于构造函数,开始会话的时候会调用,比如使用session_start()函数之后

close(); //类似于类的析构函数,在write函数调用之后调用,session_write_close()之后之后也会执行

read(string $sessionId); //读取session的时候调用

write(string $sessionId, string $data); //保存数据的时候调用

destory($sessionId); //销毁会话的时候(session_destory()或者session_regenerate_id())会调用

gc($lifeTime); //垃圾清理函数,清理掉过期作废的数据

主要就是实现这几个方法,根据不同的存储驱动可以自己设置不同的具体方法,我实现了mysql数据库和redis这两种保存session的驱动,如果有需要的话可以自己去扩展,扩展很方便很容易。

下面是我的redis的实现(db和redis差不多,redis代码少,贴出来):

我使用了接口的方式,这样扩展起来更方便,那天想用memcached了,直接添加就行了

<?php
include_once __DIR__."/interfaceSession.php";
/**
 * 以db的方式存储session
 */
class redisSession implements interfaceSession{
  /**
   * 保存session的数据库表的信息
   */
  private $_options = array(
    ‘handler‘ => null, //数据库连接句柄
    ‘host‘ => null,
    ‘port‘ => null,
    ‘lifeTime‘ => null,
  );
  /**
   * 构造函数
   * @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){
    return $this->_options[‘handler‘]->get($sessionId);
  }
  /**
   * 写入或者修改session数据
   * @param $sessionId 要写入数据的session对应的id
   * @param $sessionData 要写入的数据,已经序列化过了
   */
  public function write($sessionId, $sessionData){
    return $this->_options[‘handler‘]->setex($sessionId, $this->_options[‘lifeTime‘], $sessionData);
  }
  /**
   * 主动销毁session会话
   * @param $sessionId 要销毁的会话的唯一id
   */
  public function destory($sessionId){
    return $this->_options[‘handler‘]->delete($sessionId) >= 1 ? true : false;
  }
  /**
   * 清理绘画中的过期数据
   * @param 有效期
   */
  public function gc($lifeTime){
    //获取所有sessionid,让过期的释放掉
    $this->_options[‘handler‘]->keys("*");
    return true;
  }
}

看看简单工厂模式

class session {
    /**
     * 驱动程序句柄保存
     */
    private static $_handler = null;
    /**
     * 创建session驱动程序
     */
    public static function getSession($type, $options){
  //单例
  if(isset($handler)){
      return self::$_handler;
  }
  switch ($type) {
      case ‘db‘: //数据库驱动session类型
        include_once __DIR__."/driver/dbSession.php";
        $handler = new dbSession($options);
    break;
      case ‘redis‘: //redis驱动session类型
        include_once __DIR__."/driver/redisSession.php";
        $handler = new redisSession($options);
    break;
      default:
        return false;
    break;
  }
  return self::$_handler = $handler;
    }
}

调用也很简单,

session::getSession(‘redis‘,array(
        ‘host‘ => "localhost",
        ‘port‘ => "6379",
    ))->begin();

session_start();
时间: 2024-10-26 08:26:59

session放入缓存(redis)、DB的相关文章

将Session放入Redis

默认情况下,我们的PHP是以文件的形式保存Session数据,所以,每次读写会话信息,就需要去访问硬盘. 为了解决会话信息夸域名问题,即为了实现同一时刻只能一个地方登录,同时也解决读写会话信息必须访问磁盘问题,我想到了,将Session保存到Redis中. 下面贴代码: sessionRedis::setSessionHandler(); class sessionRedis{ public static $redis; public static function sessionOpen($s

git如何删除已经 add 的文件 (如何撤销已放入缓存区文件的修改)

使用 git rm 命令即可,有两种选择, 一种是 git rm –cached "文件路径",不删除物理文件,仅将该文件从缓存中删除: 一种是 git rm –f "文件路径",不仅将该文件从缓存中删除,还会将物理文件删除(不会回收到垃圾桶). git –如何撤销已放入缓存区(Index区)的修改 修改或新增的文件通过 git add –all命令全部加入缓存区(index区)之后,使用 git status 查看状态 (git status -s 简单模式查看状

git --如何撤销已放入缓存区(Index区)的修改

修改或新增的文件通过 git add --all 命令全部加入缓存区(index区)之后,使用 git status 查看状态(git status -s 简单模式查看状态,第一列本地库和缓存区的差异,第二列缓存区和工作目录的差异),提示使用 git reset HEAD <file> 来取消缓存区的修改. 不添加<file>参数,撤销所有缓存区的修改. 另外可以使用 git rm --cached 文件名 ,可以从缓存区移除文件,使该文件变为未跟踪的状态,同时下次提交时从本地库中

js数据放入缓存,需要再调用

再贴代码之前先描述下,这个技术应用的场景:一个页面的http请求次数能少点就少,这样大大提高用户体验.所以再一个页面发起一个请求,把所有数据都拿到后储存在缓存里面,你想用的时候再调用出来,这个是非常好的一个做法. 下面开始讲解这个技术的应用: 这个技术主要运用到getStorage setStorage 步骤如下: //定义全局变量函数var uzStorage = function () { var ls = window.localStorage; return ls;};//定义全局变量u

分布式缓存Redis应用场景解析

Redis的应用场景非常广泛.虽然Redis是一个key-value的内存数据库,但在实际场景中,Redis经常被作为缓存来使用,如面对数据高并发的读写.海量数据的读写等.举个例子,A网站首页一天有100万人访问,其中有一个"积分商城"的板块,要直接从数据库查询,那么一天就要多消耗100万次数据库请求.如果将这些数据储存到Redis(内存)中,要用的时候,直接从内存调取,不仅可以大大节省系统直接读取磁盘来获得数据的IO开销,提高服务器的资源利用率,还能极大地提升速度.随着日益增长的用户

第十章 企业项目开发--分布式缓存Redis(2)

注意:本章代码是在上一章的基础上进行添加修改,上一章链接<第九章 企业项目开发--分布式缓存Redis(1)> 上一章说了ShardedJedisPool的创建过程,以及redis五种数据结构的第一种String类型的常用缓存操作方法.下面说余下的四种: list(有序列表) set(无序集合) sorted set(有序集合) hash 1.ssmm0-cache 1.1.RedisListUtil(有序列表工具类) 1 package com.xxx.cache.redis; 2 3 im

session保存到缓存(redis)、DB

为什么要把SESSION保存在缓存 就php来说,语言本身支持的session是以文件的方式保存到磁盘文件中,保存在指定的文件夹中,保存的路径可以在配置文件中设置或者在程序中使用函数session_save_path()进行设置,但是这么做有弊端, 第一就是保存到文件系统中,效率低,只要有用到session就会从好多个文件中查找指定的sessionid,效率很低. 第二就是当用到多台服务器的时候可能会出现,session丢失问题(其实是保存在了其他服务器上). 当然了,保存在缓存中可以解决上面的

C# Azure 存储-分布式缓存Redis在session中的配置

1. 开始 对于分布式的缓存,平常的session的处理是一个用户对应一台分布式的机器,如果这台机器中途挂机或者不能处理这个用户session的情况发生,则此用户的session会丢失,会发生不可预知的错误.如下图: 如果用Redis的分布式缓存,则能避免上面的情况.因为session是保存在Redis中,不会有丢失的情况,就算中途有服务器A挂掉.如下图: 2. 代码 1)在包管理器中,输入下面的包,安装 Install-Package Microsoft.Web.RedisSessionSta

tomcat redis session共享(包含redis安全设置)

一.redis安装(我的环境centos6.5) 已安装可以跳过 1.下载:http://download.redis.io/releases/redis-2.8.19.tar.gz 2.编译源程序: 解压:tar zxvf redis-2.8.19.tar.gz tar zxvf redis-2.8.19.tar.gz cd redis-2.8.19/src 编译:如果没有gcc就执行一下 yum install gcc gcc-c++ -y make .......省略编译过程,内容太多了.