php

位置:IT落伍者 >> php >> 浏览文章

基于php使用memcache存储session的详解


发布日期:2018年05月20日
 
基于php使用memcache存储session的详解

web服务器的php session都给memcached 这样你不管分发器把 ip连接分给哪个web服务器都不会有问题了配置方法很简单就在php的配置文件内
增加一条语句就可以了不过前提你需要装好memcache模块

设置session用memcache来存储
方法I: 在 phpini 中全局设置
sessionsave_handler = memcache
sessionsave_path = "tcp://:"
方法II 某个目录下的 htaccess
php_value sessionsave_handler "memcache"
php_value sessionsave_path  "tcp://:"
方法III 再或者在某个一个应用中
ini_set("sessionsave_handler" "memcache");
ini_set("sessionsave_path" "tcp://:");
使 用多个 memcached server 时用逗号""隔开并且和 Memcache::addServer()  文档中说明的一样可以带额外的参数"persistent""weight""timeout""retry_interval"  等等类似这样的"tcp://host:port?persistent=&weight=tcp://host:port"  
如果安装的PECL是memcached(使用libmemcache库的那个)则配置应为
ini_set("sessionsave_handler" "memcached"); // 是memcached不是memcache
ini_set("sessionsave_path" ":"); // 不要tcp:

启动 memcached
memcached d l p m
或 启动Memcache的服务器端
memcached d m u root l p c P /tmp/memcachedpid  
# /usr/local/bin/memcached d m u root l p c P /tmp/memcachedpid
引用
    d选项是启动一个守护进程
m是分配给Memcache使用的内存数量单位是MB我这里是MB
u是运行Memcache的用户我这里是root
l是监听的服务器IP地址如果有多个地址的话我这里指定了服务器的IP地址
p是设置Memcache监听的端口我这里设置了最好是以上的端口我们这里统一使用
c选项是最大运行的并发连接数默认是我这里设置了按照你服务器的负载量来设定
P是设置保存Memcache的pid文件我这里是保存在/tmp/memcachedpid

在程序中使用 memcache 来作 session 存储
用例子测试一下

复制代码 代码如下:
<?php  
session_start();  
if (!isset($_SESSION[TEST])) {  
$_SESSION[TEST] = time();  
}  

$_SESSION[TEST] = time();  

print $_SESSION[TEST];  
print "<br><br>";  
print $_SESSION[TEST];  
print "<br><br>";  
print session_id();  
?>  


用 sessionid 去 memcached 里查询一下

复制代码 代码如下:
<?php  
$memcache = memcache_connect(localhost );  
var_dump($memcache>get(ccedecbceebe));  
$memcache>set(aaaa hello everyone);  
var_dump($memcache>get(aaaa));  
?>


会看到
string() "TEST|i:;TEST|i:;"
这样的输出证明 session 正常工作
用  memcache 来存储 session 在读写速度上会比 files 时快很多而且在多个服务器需要共用 session  时会比较方便将这些服务器都配置成使用同一组 memcached 服务器就可以减少了额外的工作量缺点是 session 数据都保存在  memory 中持久化方面有所欠缺但对 session 数据来说也不是很大的问题
===================================
一 般地 Session 是以文本文件形式存储在服务器端的如果使用 Seesion或者该 PHP 文件要调用 Session  变量那么就必须在调用 Session 之前启动它使用 session_start() 函数其它都不需要你设置了PHP 自动完成  Session 文件的创建其默认 Session 的存放路径是服务器的系统临时文件夹
但是如果碰到大数据量的Sesstion的时候   使用基于文件的Session存取瓶颈可能都是在磁盘IO操作上现在利用Memcached来保存Session数据直接通过内存的方式效率自然能 够提高不少 在读写速度上会比 files 时快很多而且在多个服务器需要共用 session 时会比较方便将这些服务器都配置成使用同一组  memcached 服务器就可以减少了额外的工作量

其缺点是 session 数据都保存在 memory 中一旦宕机数据将会丢失但对 session 数据来说并不是严重的问题
如何用 memcached 来存储 session呢?以下是基本的配置步骤
安装 memcached
在 phpinfo 输出中的 “Registered save handlers” 会有 “files user sqlite”

修改配置文件
a 在 phpini 中全局设置(* 需要重启服务器)
sessionsave_handler = memcache
sessionsave_path = "tcp://:"
b 或者某个目录下的 htaccess :
php_value sessionsave_handler "memcache"
php_value sessionsave_path "tcp://:"
c 也可以在某个一个应用中
ini_set("sessionsave_handler" "memcache");
ini_set("sessionsave_path" "tcp://:");
使 用多个 memcached server 时用逗号””隔开并且和 Memcache::addServer()  文档中说明的一样可以带额外的参数”persistent””weight””timeout””retry_interval”  等等类似这样的”tcp://host:port?persistent=&weight=tcp://host :port

启动 memcached
memcached d m u root l p c P /tmp/memcachedpid

测试 创建一个 session

复制代码 代码如下:
<?php
//set_sessionphp
session_start();
if (!isset($_SESSION[admin])) {
$_SESSION[TEST] = wan;
}
print $_SESSION[admin];
print "n";
print session_id();
?>


用 sessionid 去 memcached 里查询一下

复制代码 代码如下:
<?php
//get_sessionphp
$mem = new Memcache;
$mem>connect("" );
var_dump($mem>get(dbcddfefbaffa ));
?>复制代码 代码如下:
[root@localhost html]# /usr/local/webserver/php/bin/php f get_sessionphp


输出结果
string()
"admin|s::"wan";"
证明 session 正常工作
===========================
用  memcache 来存储 session 在读写速度上应该会比文件快很多而且在多个服务器需要共用 session  时会比较方便将这些服务器都配置成使用同一组 memcached 服务器就可以减少了额外的工作量缺点是 session  数据都保存在内存中不能持久化存储如果想持久化存储可以考虑使用Memcachedb来存储或用Tokyo Tyrant+Tokyo  Cabinet来进行存储

怎样判断session失效了呢?在phpini中有个Sessioncookie_lifetime的 选项这个代表SessionID在客户端Cookie储存的时间默认值是“代表浏览器一关闭SessionID就作废这样不管保存在 Memcached中的Session是否还有效(保存在Memcached中的session会利用Memcached的内部机制进行处理即使 session数据没有失效而由于客户端的SessionID已经失效所以这个key基本上不会有机会使用了利用Memcached的LRU原则 如果Memcached的内存不够用了新的数据就会取代过期以及最老的未被使用的数据)因为SessionID已经失效了所以在客户端会重新生成一 个新的SessionID

保存在Memcached中的数据最长不会超过这个时间是以操作Memcached的时间为基准的 也就是说只要key还是原来的key如果你重新对此key进行了相关的操作(如set操作)且重新设置了有效期则此时此key对应的数据的有效期 会重新计算的php手册中有说明

Expiration time of the item If its equal to  zero the item will never expire You can also use Unix timestamp or a  number of seconds starting from current time but in the latter case the  number of seconds may not exceed ( days)

Memcached主要的cache机制是LRU(最近最少用)算法+超时失效当您存数据到memcached中可以指定该数据在缓存中可以呆多久如果memcached的内存不够用了过期的slabs会优先被替换接着就轮到最老的未被使用的slabs
===========================
为了使web应用能使用saas模式的大规模访问必须实现应用的集群部署要实现集群部署主要需要实现session共享机制使得多台应用服务器之间会话统一 tomcat等多数服务都采用了session复制技术实现session的共享
session复制技术的问题:
()技术复杂必须在同一种中间件之间完成(如:tomcattomcat之间)
()在节点持续增多的情况下session复制带来的性能损失会快速增加特别是当session中保存了较大的对象而且对象变化较快时性能下降更加显着这种特性使得web应用的水平扩展受到了限制

session 共享的另一种思路就是把session集中起来管理首先想到的是采用数据库来集中存储session但数据库是文件存储相对内存慢了一个数量级同时 这势必加大数据库系统的负担所以需要一种既速度快又能远程集中存储的服务所以就想到了memcached

memcached能缓存什么?
通过在内存里维护一个统一的巨大的hash表Memcached能够用来存储各种格式的数据包括图像视频文件以及数据库检索的结果等

memcached快么?
非 常快memcached使用了libevent(如果可以的话在linux下使用epoll)来均衡任何数量的打开链接使用非阻塞的网络I/O对 内部对象实现引用计数(因此针对多样的客户端对象可以处在多样的状态) 使用自己的页块分配器和哈希表  因此虚拟内存不会产生碎片并且虚拟内存分配的时间复杂度可以保证为O()
使用过程注意几个问题和改进思路
memcache的内存应该足够大这样不会出现用户session从Cache中被清除的问题(可以关闭memcached的对象退出机制)
如果session的读取比写入要多很多可以在memcache前再加一个Oscache等本地缓存减少对memcache的读操作从而减小网络开销提高性能
如果用户非常多可以使用memcached组通过set方法中带hashCode插入到某个memcached服务器
对于session的清除有几种方案:
()可以在凌晨人最少的时候对memcached做一次清空(简单)
()保存在缓存中的对象设置一个失效时间通过过滤器获取sessionId的值定期刷新memcached中的对象长时间没有被刷新的对象自动被清除(相对复杂消耗资源)

               

上一篇:PHP的可变变量名

下一篇:一个简单的php在线端口扫描器