解析php中session的实现原理以及大网站应用应注意的问题


Posted in PHP onJune 17, 2013

PHP SESSION原理
我们知道,session是在服务器端保持用户会话数据的一种方法,对应的cookie是 在客户端保持用户数据。HTTP协议是一种无状态协议,服务器响应完之后就失去了与浏览器的联系,最早,Netscape将cookie引入浏览器,使得 数据可以客户端跨页面交换,那么服务器是如何记住众多用户的会话数据呢?

首先要将客户端和服务器端建立一一联系,每个客户 端都得有一个唯一标识,这样服务器才能识别出来。建议唯一标识的方法有两种:cookie或者通过GET方式指定。默认配置的PHP使用session的 时会建立一个名叫”PHPSESSID”的cookie(可以通过php.ini修改session.name值指定),如果客户端禁用cookie,你 也可以指定通过GET方式把session id传到服务器(修改php.ini中session.use_trans_sid等参数)。

我们查看服务器端session.save_path目录会发现很多类似sess_vv9lpgf0nmkurgvkba1vbvj915这样的文件,这个 其实就是session id “vv9lpgf0nmkurgvkba1vbvj915″对应的数据。真相就在这里,客户端将session id传递到服务器,服务器根据session id找到对应的文件,读取的时候对文件内容进行反序列化就得到session的值,保存的时候先序列化再写入。

事实就是这 样,所以如果服务器不支持session或者你想自定义session,完全可以DIY,通过PHP的uniqid生成永不重复的session id,然后找个地方存储session的内容即可,你也可以学flickr把session存储在MySQL数据库中。

使用session之前为什么必须先执行session_start()?
了 解的原理之后,所谓的session其实就是客户端一个session id服务器端一个session file,新建session之前执行session_start()是告诉服务器要种一个cookie以及准备好session文件,要不然你的 session内容怎么存;读取session之前执行session_start()是告诉服务器,赶紧根据session id把session文件反序列化。

只有一个session函数可以在session_start()之前执行,session_name():读取或指定session名称(比如默认的就是”PHPSESSID”),这个当然要在session_start之前执行。

session影响系统性能
session 在大访问量网站上确实影响系统性能,影响性能的原因之一由文件系统设计造成,在同一个目录下超过10000个文件时,文件的定位将非常耗时,PHP支持 session目录hash,我们可以通过修改php.ini中session.save_path = “2;/path/to/session/dir”,那么session将存储在两级子目录中,每个目录有16个子目录[0~f],不过好像PHP session不支持创建目录,你需要事先把那么些目录创建好 。

还有一个问题就是小文件的效率问题,一般我们的 session数据都不会太大(1~2K),如果有大量这样1~2K的文件在磁盘上,IO效率肯定会很差,PHP手册上建议使用Reiserfs文件系 统,不过Reiserfs的前景堪忧,Reiserfs的作者把媳妇给杀了,SuSE也抛弃了Reiserfs。

其实还有很多中 存储session的方式,可以通过php -i|grep “Registered save handlers”查看,比如Registered save handlers => files user sqlite eaccelerator可以通过文件、用户、sqlite、eaccelerator来存,如果服务器装了memcached,还有会mmcache的 选项。当然还有很多,比如MySQL、PostgreSQL等等。都是不错的选择。

session的同步
我们前端可能有很多台服务器,用户在A服务器上登录了,种下了session信息,然后访问网站的某些页面没准跳到B服务器上去了,如果这个时候B服务器上没有session信息又没有做特殊处理,可能就会出问题了。
session同步有很多种,如果你是存储在memcached或者MySQL中,那就很容易了,指定到同样的位置即可,如果是文件形式的,你可以用NFS统一存储。

还有一种方式是通过加密的cookie来实现,用户在A服务器上登录成功,在用户的浏览器上种上一个加密的cookie,当用户访问B服务器时,检查有无 session,如果有当然没问题,如果没有,就去检验cookie是否有效,cookie有效的话就在B服务器上重建session。这种方法其实很有 用,如果网站有很多个子频道,服务器也不在一个机房,session没办法同步又想做统一登录那就太有用了。

当然还有一种方法就 是在负载均衡那一层保持会话,把访问者绑定在某个服务器上,他的所有访问都在那个服务器上就不需要session同步了,这些都是运维层面的东西。就说这 么多吧,根据自己的应用来选择使用session,不要因为大家都说session影响系统性能就畏首畏尾,知道问题,解决问题才是关键,惹不起躲得起不适合这里。

PHP 相关文章推荐
用PHP实现的随机广告显示代码
Jun 14 PHP
php feof用来识别文件末尾字符的方法
Aug 01 PHP
使用PHP遍历文件夹与子目录的函数代码
Sep 26 PHP
php中使用DOM类读取XML文件的实现代码
Dec 14 PHP
解决PHP mysql_query执行超时(Fatal error: Maximum execution time …)
Jul 03 PHP
destoon二次开发模板及调用语法汇总
Jun 21 PHP
PHP中file_exists()判断中文文件名无效的解决方法
Nov 12 PHP
浅析php适配器模式(Adapter)
Nov 25 PHP
php使用glob函数遍历文件和目录详解
Sep 23 PHP
Yii框架用户登录session丢失问题解决方法
Jan 07 PHP
Paypal实现循环扣款(订阅)功能
Mar 23 PHP
PHP实现UTF8二进制及明文字符串的转化功能示例
Nov 20 PHP
CodeIgniter图像处理类的深入解析
Jun 17 #PHP
jQuery+php实现ajax文件即时上传的详解
Jun 17 #PHP
深入Apache与Nginx的优缺点比较详解
Jun 17 #PHP
php之CodeIgniter学习笔记
Jun 17 #PHP
apache配置虚拟主机的方法详解
Jun 17 #PHP
php unset全局变量运用问题的深入解析
Jun 17 #PHP
Linux Apache PHP Oracle 安装配置(具体操作步骤)
Jun 17 #PHP
You might like
一家之言的经验之谈php+mysql扎实个人基本功
2008/03/27 PHP
浅析php适配器模式(Adapter)
2014/11/25 PHP
php使用递归函数实现数字累加的方法
2015/03/16 PHP
最新版本PHP 7 vs HHVM 多角度比较
2016/02/14 PHP
Thinkphp3.2.3分页使用实例解析
2016/07/28 PHP
Laravel timestamps 设置为unix时间戳的方法
2019/10/11 PHP
JS控件autocomplete 0.11演示及下载 1月5日已更新
2007/01/09 Javascript
javascript 模式设计之工厂模式详细说明
2010/05/10 Javascript
jquery isEmptyObject判断是否为空对象的函数
2011/02/14 Javascript
加载 Javascript 最佳实践
2011/10/30 Javascript
解决wx.onMenuShareTimeline出现的问题
2016/08/16 Javascript
JavaScript模仿Pinterest实现图片预加载功能
2016/10/25 Javascript
微信小程序开发(二)图片上传+服务端接收详解
2017/01/11 Javascript
微信小程序 常见问题总结(4058,40013)及解决办法
2017/01/11 Javascript
深入理解Nodejs Global 模块
2017/06/03 NodeJs
Vue下拉框回显并默认选中随机问题
2018/09/06 Javascript
前端防止用户重复提交js实现代码示例
2018/09/07 Javascript
浅谈vue加载优化策略
2019/03/19 Javascript
layui table 表格模板按钮的实例代码
2019/09/21 Javascript
[07:31]DOTA2卡尔工作室 英雄介绍主宰篇
2013/06/25 DOTA
[06:33]DOTA2亚洲邀请赛小组赛第二日 TOP10精彩集锦
2015/01/31 DOTA
python3利用Dlib19.7实现人脸68个特征点标定
2018/02/26 Python
python实现在图片上画特定大小角度矩形框
2018/10/24 Python
Python实现搜索算法的实例代码
2020/01/02 Python
python+selenium+chromedriver实现爬虫示例代码
2020/04/10 Python
Python爬虫中Selenium实现文件上传
2020/12/04 Python
沙特阿拉伯网上购物:Sayidaty Mall
2018/05/06 全球购物
经济学人订阅:The Economist
2018/07/19 全球购物
乡镇信息公开实施方案
2014/03/23 职场文书
经典演讲稿开场白
2014/08/25 职场文书
2014年小学德育工作总结
2014/12/05 职场文书
大学生英文求职信范文
2015/03/19 职场文书
个人求职信格式范文
2015/03/20 职场文书
幼儿园门卫安全责任书
2015/05/08 职场文书
浅谈Python列表嵌套字典转化的问题
2021/04/07 Python
MySQL中的 inner join 和 left join的区别解析(小结果集驱动大结果集)
2023/05/08 MySQL