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 相关文章推荐
SMARTY学习手记
Jan 04 PHP
PHP中实现汉字转区位码应用源码实例解析
Jun 14 PHP
jQuery+PHP+ajax实现微博加载更多内容列表功能
Jun 27 PHP
让ThinkPHP支持大小写url地址访问的方法
Oct 31 PHP
php启用sphinx全文搜索的实现方法
Dec 24 PHP
php二维码生成
Oct 19 PHP
简单实现PHP留言板功能
Dec 21 PHP
基于thinkPHP类的插入数据库操作功能示例
Jan 06 PHP
PHP实现驼峰样式字符串(首字母大写)转换成下划线样式字符串的方法示例
Aug 10 PHP
详解php伪造Referer请求反盗链资源
Jan 24 PHP
php5.6.x到php7.0.x特性小结
Aug 17 PHP
laravel实现上传图片并在页面显示的例子
Oct 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
全国FM电台频率大全 - 11 浙江省
2020/03/11 无线电
dedecms防止FCK乱格式化你的代码的修改方法
2007/03/17 PHP
PHP 5.3新特性命名空间规则解析及高级功能
2010/03/11 PHP
解析mysql left( right ) join使用on与where筛选的差异
2013/06/18 PHP
Laravel + Elasticsearch 实现中文搜索的方法
2020/02/02 PHP
在IE下:float属性会影响offsetTop的取值
2006/12/22 Javascript
javascript编程起步(第一课)
2007/01/10 Javascript
jquery 插件学习(五)
2012/08/06 Javascript
JavaScript实现彩虹文字效果的方法
2015/04/16 Javascript
AngularJS监听路由的变化示例代码
2016/09/23 Javascript
Node.js用readline模块实现输入输出
2016/12/16 Javascript
Vue.js 2.0 移动端拍照压缩图片预览及上传实例
2017/04/27 Javascript
Vue-cli Eslint在vscode里代码自动格式化的方法
2018/02/23 Javascript
p5.js入门教程之平滑过渡(Easing)
2018/03/16 Javascript
详解js中Array的方法及技巧
2018/09/12 Javascript
基于vue实现圆形菜单栏组件
2019/07/05 Javascript
nodejs二进制与Buffer的介绍与使用
2019/07/11 NodeJs
Angular8路由守卫原理和使用方法
2019/08/29 Javascript
python下载文件时显示下载进度的方法
2015/04/02 Python
python并发编程之线程实例解析
2017/12/27 Python
Python实现的径向基(RBF)神经网络示例
2018/02/06 Python
PyQt5 窗口切换与自定义对话框的实例
2019/06/20 Python
python实现图像检索的三种(直方图/OpenCV/哈希法)
2019/08/08 Python
Django-rest-framework中过滤器的定制实例
2020/04/01 Python
Django+Celery实现动态配置定时任务的方法示例
2020/05/26 Python
基于Python正确读取资源文件
2020/09/14 Python
Python3+SQLAlchemy+Sqlite3实现ORM教程
2021/02/16 Python
HTML5如何使用SVG的方法示例
2019/01/11 HTML / CSS
廉价连衣裙和婚纱礼服在线销售:Tbdress
2019/02/28 全球购物
香港演唱会订票网站:StubHub香港
2019/10/10 全球购物
毕业生怎样写好自荐信
2013/11/11 职场文书
公司财务流程之主管工作流程
2014/03/03 职场文书
2014标准社保办理委托书
2014/10/06 职场文书
2014年政工师工作总结
2014/12/18 职场文书
php随机生成验证码,php随机生成数字,php随机生成数字加字母!
2021/04/01 PHP
Golang 实现 WebSockets 之创建 WebSockets
2022/04/24 Golang