php中使用session_set_save_handler()函数把session保存到MySQL数据库实例


Posted in PHP onNovember 06, 2014

PHP保存session默认的是采用的文件的方式来保存的,这仅仅在文件的空间开销很小的windows上是可以采用的,但是如果我们采用uinx或者是liux上的文件系统的时候,这样的文件系统的文件空间开销是很大的,然而session是要时时刻刻的使用的,大量的用户就要创建很多的session文件,这样对整个的服务器带来性能问题。

另一方面,如果服务器起采用群集的方式的话就不能保持session的一致性,所以我们就绪要采用数据库的方式来保存session,这样,不管有几台服务器同时使用,只要把他们的session保存在一台数据库服务器上就可以保证session的完整了,具体如何来实现请继续看下去。

PHP保存session默认的情况下是采用的文件方式来保存的,我们在PHP的配制文件PHP.ini中可以看到这样的一行:

session.save_handler="files"

这样的意思就是采用文件来保存session 的,要采用数据库来保存的话,我们需要修改成用户模式,改成
session.save_handler="use"

就可以了,但是,这仅仅是说明我门没有采用文件的方式存储session,我们还要选择数据库和建立数据库的表。

建立数据库和数据库的表结构,我们可以采用PHP可以使用的任何的数据库,因为PHP和mysql的结合最好,我就使用mysql来做示例,当然根据你的需要可以改称别的数据库。

创建数据库

create database 'session';

创建表结构
create table 'session'( id char(32) not null , 'user 'char(30), data char(3000) ,primary key ('id') );

PHP保存session编写PHP文件
<?php

$con = mysql_connect("127.0.0.1", "user" , "pass");

mysql_select_db("session");

function open($save_path, $session_name) {

 return(true);

}

function close() {

 return(true);

}

function read($id) {

 if ($result = mysql_query("select * from session where id='$id'")) {

  if ($row = mysql_felth_row($result)) {

   return $row["data"];

  }

 } else {

  return "";

 }

}

function write($id, $sess_data) {

 if ($result = mysql_query("update session set data='$sess_data' where id='$id'")) {

  return true;

 } else {

  return false;

 }

}

function destroy($id) {

 if ($result = mysql_query("delete * from session where id='$id'")) {

  return true;

 } else {

  return false;

 }

}

function gc($maxlifetime) {

 return true;

}

session_set_save_handler("open", "close", "read", "write", "destroy", "gc");

session_start();

// proceed to use sessions normally

保存成为session_user_start.php。

现在我们的PHP保存session的工作就已经完成了,只要你在需要在使用session的时候,把session_user_start.php包含进来.注意,这个文件一定要在文件的第一行包含,然后就像使用文件的session一样的方法使用就可以了。

以上仅仅是个简单教程,在实际的应用中,可以对它封装得更专业些,参考代码如下:

SessionMysql.class.php

<?php

/**

 * SessionMysql 数据库存储类

 */
defined('IN_QIAN') or exit('Access Denied');
class SessionMysql {
 public $lifetime = 1800; // 有效期,单位:秒(s),默认30分钟

 public $db;

 public $table;
 /**

  * 构造函数

  */

 public function __construct() {

  $this->db = Base::loadModel('SessionModel');

  $this->lifetime = Base::loadConfig('system', 'session_lifetime');

  session_set_save_handler(

   array(&$this, 'open'),  // 在运行session_start()时执行

   array(&$this, 'close'),  // 在脚本执行完成 或 调用session_write_close() 或 session_destroy()时被执行,即在所有session操作完后被执行

   array(&$this, 'read'),  // 在运行session_start()时执行,因为在session_start时,会去read当前session数据

   array(&$this, 'write'),  // 此方法在脚本结束和使用session_write_close()强制提交SESSION数据时执行

   array(&$this, 'destroy'), // 在运行session_destroy()时执行

   array(&$this, 'gc')   // 执行概率由session.gc_probability 和 session.gc_divisor的值决定,时机是在open,read之后,session_start会相继执行open,read和gc

  );

  session_start(); // 这也是必须的,打开session,必须在session_set_save_handler后面执行

 }

 /**

  * session_set_save_handler open方法

  *

  * @param $savePath

  * @param $sessionName

  * @return true

  */

 public function open($savePath, $sessionName) {

  return true;

 }

 /**

  * session_set_save_handler close方法

  *

  * @return bool

  */

 public function close() {

  return $this->gc($this->lifetime);

 }

 /**

  * 读取session_id

  *

  * session_set_save_handler read方法

  * @return string 读取session_id

  */

 public function read($sessionId) {

  $condition = array(

   'where' => array(

    'session_id' => $sessionId

   ),

   'fields' => 'data'

  );

  $row = $this->db->fetchFirst($condition);

  return $row ? $row['data'] : '';

 }

 /**

  * 写入session_id 的值

  *

  * @param $sessionId 会话ID

  * @param $data 值

  * @return mixed query 执行结果

  */

 public function write($sessionId, $data) {

  $userId = isset($_SESSION['userId']) ? $_SESSION['userId'] : 0;

  $roleId = isset($_SESSION['roleId']) ? $_SESSION['roleId'] : 0;

  $grouId = isset($_SESSION['grouId']) ? $_SESSION['grouId'] : 0;

  $m = defined('ROUTE_M') ? ROUTE_M : '';

  $c = defined('ROUTE_C') ? ROUTE_C : '';

  $a = defined('ROUTE_A') ? ROUTE_A : '';

  if (strlen($data) > 255) {

   $data = '';

  }

  $ip = get_ip();

  $sessionData = array(

   'session_id' => $sessionId,

   'user_id'  => $userId,

   'ip'   => $ip,

   'last_visit' => SYS_TIME,

   'role_id'  => $roleId,

   'group_id'  => $grouId,

   'm'    => $m,

   'c'    => $c,

   'a'    => $a,

   'data'   => $data,

  );

  return $this->db->insert($sessionData, 1, 1);

 }

 /**

  * 删除指定的session_id

  *

  * @param string $sessionId 会话ID

  * @return bool

  */

 public function destroy($sessionId) {

  return $this->db->delete(array('session_id' => $sessionId));

 }

 /**

  * 删除过期的 session

  *

  * @param $lifetime session有效期(单位:秒)

  * @return bool

 */

 public function gc($lifetime) {

  $expireTime = SYS_TIME - $lifetime;

  return $this->db->delete("`last_visit`<$expireTime");

 }

}

在系统文件的某个地方,实例化这个类即可!

new SessionMysql();
PHP 相关文章推荐
如何在PHP中进行身份认证
Oct 09 PHP
关于PHP中Object对象的笔记分享
Jun 28 PHP
PHP动态创建Web站点的方法
Aug 14 PHP
PHP常用特殊运算符号和函数总结(php新手入门必看)
Feb 02 PHP
PHP多线程批量采集下载美女图片的实现代码(续)
Jun 03 PHP
php二分查找二种实现示例
Mar 12 PHP
yii实现CheckBox复选框在同一行显示的方法
Dec 03 PHP
ThinkPHP3.2.2实现持久登录(记住我)功能的方法
May 16 PHP
Yii 2.0实现联表查询加搜索分页的方法示例
Aug 02 PHP
bindParam和bindValue的区别以及在Yii2中的使用详解
Mar 12 PHP
PHP错误提示It is not safe to rely on the system……的解决方法
Mar 25 PHP
laravel实现图片上传预览,及编辑时可更换图片,并实时变化的例子
Nov 14 PHP
php中常见的sql攻击正则表达式汇总
Nov 06 #PHP
php中实现记住密码下次自动登录的例子
Nov 06 #PHP
php网站被挂木马后的修复方法总结
Nov 06 #PHP
调试PHP程序的多种方法介绍
Nov 06 #PHP
php实现singleton()单例模式实例
Nov 06 #PHP
php使用session二维数组实例
Nov 06 #PHP
php函数serialize()与unserialize()用法实例
Nov 06 #PHP
You might like
PHP Token(令牌)设计
2008/03/15 PHP
关于UEditor编辑器远程图片上传失败的解决办法
2012/08/31 PHP
PHP实现读取一个1G的文件大小
2013/08/24 PHP
关于PHP 如何用 curl 读取 HTTP chunked 数据
2016/02/26 PHP
PHP匿名函数和use子句用法实例
2016/03/16 PHP
PHP 序列化和反序列化函数实例详解
2020/07/18 PHP
ubutu 16.04环境下,PHP与mysql数据库,网页登录验证实例讲解
2017/07/20 PHP
JQuery live函数
2010/12/24 Javascript
js解析与序列化json数据(一)json.stringify()的基本用法
2013/02/01 Javascript
有关于JS辅助函数inherit()的问题
2013/04/07 Javascript
JavaScript中textRange对象使用方法小结
2015/03/24 Javascript
this,this,再次讨论javascript中的this,超全面(经典)
2016/01/05 Javascript
jquery无法为动态生成的元素添加点击事件的解决方法(推荐)
2016/12/26 Javascript
基于Vue开发数字输入框组件
2017/12/19 Javascript
js实现滑动进度条效果
2020/08/21 Javascript
element-ui中el-upload多文件一次性上传的实现
2020/12/02 Javascript
[00:35]DOTA2上海特级锦标赛 MVP.Phx战队宣传片
2016/03/04 DOTA
python 实现一个贴吧图片爬虫的示例
2017/10/12 Python
Python Json序列化与反序列化的示例
2018/01/31 Python
Python中property属性实例解析
2018/02/10 Python
Python 字符串操作(string替换、删除、截取、复制、连接、比较、查找、包含、大小写转换、分割等)
2018/03/19 Python
在Python文件中指定Python解释器的方法
2019/02/18 Python
Python中函数参数匹配模型详解
2019/06/09 Python
python matplotlib画盒图、子图解决坐标轴标签重叠的问题
2020/01/19 Python
Django关于admin的使用技巧和知识点
2020/02/10 Python
tensorboard 可以显示graph,却不能显示scalar的解决方式
2020/02/15 Python
意大利网上药房:Farmacia 33
2020/01/27 全球购物
党的群众路线教育实践活动动员会主持词
2014/03/20 职场文书
经济担保书范文
2014/04/02 职场文书
宿舍标语大全
2014/06/19 职场文书
2014年文明创建工作总结
2014/11/25 职场文书
委托公证书样本
2015/01/23 职场文书
警告通知
2015/04/25 职场文书
pytorch 两个GPU同时训练的解决方案
2021/06/01 Python
仅仅使用 HTML/CSS 实现各类进度条的方式汇总
2021/11/11 HTML / CSS
Vue的列表之渲染,排序,过滤详解
2022/02/24 Vue.js