PHP中多服务器共享SESSION的方法

在PHP中SESSION默认是用文件来进行保存的,如session.save_path=/tmp,则是将session数据保存在/tmp目录下,如果并发用户量很大,在这个目录下就会存在大量类似sess_xxxxxx的session文件,导致 性能 下降,不少朋友可能都没有注意到php.ini里面Session设置部分中有这样一项:

  ; session.save_path = "N; MODE; /path"

  这项设置提供给我们可以给session存放目录进行多级散列,其中“N”表示要设置的目录级数,“MODE”表示目录的权限属性,默认为600,在WINDOWS上基本是不用设置的,*NIX上也可以不用设置,后面的“/path”表示session文件存放的根目录路径,比如我们设置为下面的格式

  session.save_path = "2; d:/php5/tmp"

  上面的设置表示我们把d:/php5/tmp目录作为php的session文件存放根目录,在该目录下进行两级目录散列,每一级目录分别是0-9和a-z共36个字母数字为目录名,这样存放session的目录可以达到36*36个,相信作为单台服务器来说,这是完全够用了,如果说您的系统架构设计为多台服务器共享session数据,可以把目录级增加到3级或者更多。

  需要注意的是,php自己并不会自动创建子目录,需要您自己动手去创建,网上找到这样的自动创建目录的代码,大家可以做个参考。下面的代码自动创建3级子目录,可以自己动手根据需要进行修改。

<?php
set_time_limit(0); 
$string = '0123456789abcdefghijklmnopqrstuvwxyz'; 
$length = strlen($string); 
function makeDir($param)
{
if(!file_exists($param)) {
makeDir(dirname($param)); 
mkdir($param); 
}
} 
for($i = 0; $i < $length; $i++) {
for($j = 0; $j < $length; $j++) {
for($k = 0; $k < $length; $k++) {
makeDir($string[$i].'/'.$string[$j].'/'.$string[$k]); 
}
}
}
?>
 

多服务器共享session的方法:

  1.通过NFS文件共享的方式,多台WEB服务器共享保存session文件的磁盘

  2.保存在数据库中,这种方式的扩展性很强,可以随意增加WEB而不受影响

  3.可以将session数据保存在memcached中,memcached是基于内存存储数据的, 性能 很高,用户并发量很大的时候尤其合适,参考PHP实现多服务器session共享之memcache共享

  4.文件方式保存session时,可以采用php的扩展eaccelerator来存储sesion,参考

  eaccelerator 应用之“使用共享内存存储Session”

  首先创建表,用来保存session数据

        CREATE TABLE `sessions` (              
      `sessionid` varchar(32) NOT NULL default '',    
      `data` mediumtext NOT NULL,            
      `lastvisit` int(10) unsigned NOT NULL default '0', 
      PRIMARY KEY (`sessionid`)             
     ) ;

主要的代码如下:

  session_mysql.class.php

<?php class session {  var $lifetime = 1800;     var $op = '';

    function __construct($op = '')     {         $this -> op = $op;      session_set_save_handler(array(&$this,'open'), array(&$this,'close'), array(&$this,'read'), array(&$this,'write'), array(&$this,'destroy'), array(&$this,'gc'));         $this->debug('session_id:' . $_COOKIE[session_name()]);         //确保读写session时session id保存在同一个域下,否则有可能会每次重新生成一个session id         ini_set('session.cookie_domain', '.chf.com');   session_start();     }

    function debug($str) {         //echo $str . "<br />\n";         $file = dirname(__FILE__) . '/session.log';         $content = date("Y-m-d H:i:s") . ' operate:' . $this -> op . ' ' . $str . "\n";         $fp = @fopen($file, "ab");         if (!$fp) die("Open file $file failed!");         @fwrite($fp, $content);         if ($fp) @fclose($fp);     }

    function session()     {   $this->__construct();     }

    /**      * 这个函数被session处理程序调用来作初始化工作。    * @param string $save_path 参数$sess_path对应php.ini文件中的session.save_path选项      * @param string $session_name 参数$session_name对应php.ini中的session.name 选项。      */     function open($save_path, $session_name)  {         $this->debug('opening...');   global $db;      $this->lifetime = 20; //session超时时间      $this->time = time();   $this->sess = &$db;         $this->pre = '';   return true;     }

    /*     sess_close();   这个函数在页面结束执行并且session处理程序需要关闭时被调用     */     function close()  {         $this->debug('closing...');   $this->gc($this->lifetime);         return $this->sess->close();     }

    //根据session id取出指定的session数据是否为空,然后判断用户登录与否或操作是否超时     /*     这个函数在session处理程序读取指定session键值($key)时,检索并返回标识为$key的session数据.(注意:序列化是将变量或对象在程序结束或需要时保存在文件中,在下次程序运行或需要时再调入内存的技术,有别于只保存数据的方法。)

    例如读session数据时会调用此函数     $name = $_SESSION['name'];     echo "name:$name<br />";

    要测试,给session赋值时也调用此函数     $_SESSION['name'] = 'caihf';

    @param string $id session id通常通过cookie来保存用户的session id,如650fcm4p8aodn1cfrusc5ehql0     */     function read($id)  {         $this->debug('reading...');         $sql = "SELECT data FROM `{$this->pre}sessions` WHERE sessionid='$id'";   $r = $this->sess->get_one($sql);         $this->debug("sql:$sql");   return $r ? $r['data'] : '';     }

    /*     这个函数据在session处理程序需要将数据保存时调用,这种情况经常在程序结束时发生。它负责将数据保存在下次能用sess_read($key)函数检索的地方。     例如给session赋值时($_SESSION['name'] = 'caihf';)会调用此函数     @param string $id session id通常通过cookie来保存用户的session id,如650fcm4p8aodn1cfrusc5ehql0     @param string $sess_data sessoin数据,一般是自动序列化以后保存在数据库中的,如name|s:5:\"caihf\";     */     function write($id, $sess_data)  {         $this->debug('writing...');         $sql = "REPLACE INTO `{$this->pre}sessions` (sessionid, data, lastvisit) VALUES('$id', '".addslashes($sess_data)."', '".time()."')";         $this->sess->query($sql);         $this->debug("sql:$sql");   return true;     }

    /*     这个函数在需要消毁session时。它负责删除session并且清除环境。     @param string $id session id通常通过cookie来保存用户的session id,如650fcm4p8aodn1cfrusc5ehql0     */     function destroy($id)  {         $this->debug('destroying...');         $sql = "DELETE FROM `{$this->pre}sessions` WHERE sessionid='$id'";   $this->sess->query($sql);         $this->debug("sql:$sql");   return true;     }

    //删除超时的session数据     /*     这个函数负责清理碎片。在这种情况下,它负责删除过时的session数据。session处理程序会偶尔调用它们。     */     function gc($maxlifetime)  {         $this->debug("gcing...\n");   $expiretime = $this->time-$maxlifetime;         $sql = "DELETE FROM `{$this->pre}sessions` WHERE lastvisit<$expiretime";   $this->sess->query($sql);         $this->debug("sql:$sql");   return true;     } }

/* 给session赋值时的log,如$_SESSION['name'] = 'caihf'; 2008-09-17 03:35:35 operate:write session_id:650fcm4p8aodn1cfrusc5ehql0 2008-09-17 03:35:35 operate:write opening... 2008-09-17 03:35:35 operate:write reading... 2008-09-17 03:35:35 operate:write sql:SELECT data FROM `sessions` WHERE sessionid='650fcm4p8aodn1cfrusc5ehql0' 2008-09-17 03:35:35 operate:write writing... 2008-09-17 03:35:35 operate:write sql:REPLACE INTO `sessions` (sessionid, data, lastvisit) VALUES('650fcm4p8aodn1cfrusc5ehql0', 'name|s:5:\"caihf\";', '1221622535') 2008-09-17 03:35:35 operate:write closing... 2008-09-17 03:35:35 operate:write gcing... 2008-09-17 03:35:35 operate:write sql:DELETE FROM `sessions` WHERE lastvisit<1221622515

从session中读数据时的log,如$name = $_SESSION['name'];echo "name:$name<br />"; 2008-09-17 03:44:24 operate:read session_id:650fcm4p8aodn1cfrusc5ehql0 2008-09-17 03:44:24 operate:read opening... 2008-09-17 03:44:24 operate:read reading... 2008-09-17 03:44:24 operate:read sql:SELECT data FROM `sessions` WHERE sessionid='650fcm4p8aodn1cfrusc5ehql0' 2008-09-17 03:44:24 operate:read writing... 2008-09-17 03:44:24 operate:read sql:REPLACE INTO `sessions` (sessionid, data, lastvisit) VALUES('650fcm4p8aodn1cfrusc5ehql0', 'name|s:5:\"caihf\";', '1221623064') 2008-09-17 03:44:24 operate:read closing... 2008-09-17 03:44:24 operate:read gcing... 2008-09-17 03:44:24 operate:read sql:DELETE FROM `sessions` WHERE lastvisit<1221623044 */ ?>

session_test1.php

<?php include_once('session_mysql.class.php'); include_once('db_mysql.class.php');

$db = new db_mysql(); $db -> connect('localhost', 'root', 'home', 'test'); $session = new session('write'); $_SESSION['name'] = 'caihf'; //echo "name:" . $_SESSION['name'] . "<br />\n"; echo 'Assign value to session over!'; ?>


session_test2.php

<?php include_once('session_mysql.class.php'); include_once('db_mysql.class.php');

$db = new db_mysql(); $db -> connect('localhost', 'root', 'home', 'test'); $session = new session('read');

function is_login() {     return strlen($_SESSION['name']) > 0 ? 1 : 0; }

if (is_login()) {     echo "Hello {$_SESSION['name']} <br />\n"; } else {     echo "User not loggined! <br />\n"; } //unset($_SESSION['name']); /* 执行完unset($_SESSION['name']);操作,然后再刷新此页面时,字段data的值就变为空了 sessionid                  data lastvisit am1dmhbd2dsq9fpglpdh81u0d6  1221634486 */ ?>



 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值