一、Session的原理
以下以默认情况举例:
session_start();之后,会生成一个唯一的session_id,每一个用户对应唯一一个session_id,每一个session_id对应服务器端的一个session文件。这个session文件存储着当前session_id的信息,比如下面,就存储了name和age的键值。
1、设置Session存储的引擎(本地服务器的文件还是redis等),【php.ini 文件】
[Session] session.save_handler = files session.save_path = /data/SessionLogs
2、默认情况下的Session的使用
<?php /** * session的使用 * 默认情况(不更改session.save_handle参数时),是存储在文件file中的 * 默认情况下使用session的情况(用户24分钟内没有刷新操作会过期) * 每个用户对应唯一session_id,每一个session_id对应服务器中存储的一个session文件,这个文件中存储了当前session_id的信息,比如下面,就存储了name和age的键值 */ session_start(); echo session_id(); echo "<br>"; $_SESSION['age'] = 26; $_SESSION['name'] = 'xiaobudiu'; var_dump($_SESSION);
3、在服务器中存储的形式是这样的
二、使用Redis存储Session
在网站访问量较大时,我们通常会做集群(比如nginx负载均衡等),这时,如何解决session会话的共享问题。
(1)使用ip_hash或者自定义key做负载均衡轮询策略是一个办法,但由于有时候用户可能走代理,所以这个方法其实并不是那么完美。
(2)另一个解决session共享问题的方法就是使用redis或者memcache缓存数据库去存储session,进而实现session共享问题。
1、设置php.ini 文件中的session.save_handle 和session.save_path
session.save_handler = Redis session.save_path = "tcp://localhost:6379"
注1:如果连接的是远程redis,需要将localhost换成对应的远程ip地址。像这样,
session.save_handler = Redis
session.save_path = "tcp://47.94.203.119:6379"
注2:如果为redis已经添加了auth权限(requirpass),session.save_path项则应该这样写
session.save_handler = Redis session.save_path = "tcp://47.94.203.119:6379?persistent=1&database=10&auth=myredisG506"
2、使用redis存储session信息
<?php /** * 将session存储在redis中 */ session_start(); echo session_id(); echo "<br>"; $_SESSION['age'] = 26; $_SESSION['name'] = 'xiaobudiu'; $_SESSION['sex'] = 'man'; var_dump($_SESSION);
在redis上是以这样的形式进行存储的
三、使用Redis存储Session,并设置Session会话存活时间以及Session中某一元素存活时间
封装session类 b.php
<?php /** * session控制类 */ class Session { function __construct($lifetime = 3600) { //初始化设置session会话存活时间 ini_set('session.gc_maxlifetime',$lifetime); } /** * 设置当前会话session的key-value * @param String $name session name * @param Mixed $data session data * @param Int $expire 有效时间(秒) */ function set($name, $data, $expire = 600) { $session_data = array(); $session_data['data'] = $data; $session_data['expire'] = time()+$expire; $_SESSION[$name] = $session_data; } /** * 读取当前会话session中的key-value * @param String $name session name * @return Mixed */ function get($name) { if(isset($_SESSION[$name])) { if($_SESSION[$name]['expire'] > time()) { return $_SESSION[$name]['data']; }else{ self::clear($name); } } return false; } /** * 清除当前session会话中的某一key-value * @param String $name session name */ function clear($name) { unset($_SESSION[$name]); } /** * 删除当前session_id对应的session文件(清空当前session会话存储) */ function destroy() { session_destroy(); } }
session类的使用:d.php
<?php require_once 'b.php'; session_start(); $session = new Session(); $session->set('wan','kkkk',1966); $session->set('name','xiaobudiu'); $session->set('age',26); $session->set('sex','man'); //输出当前会话的session存储数据 var_dump($_SESSION); //unset掉某一个session属性 //$session->clear('name'); //删除当前session_id对应session文件 //$session->destroy(); //echo $session->get('sex');
redis中显示: