Memcached 笔记与总结(9)Memcached 与 Session

一、Memcached 存储 Session

由于 Memcached 是分布式的内存对象缓存系统,因此可以用来实现 Session 同步:把 Web 服务器中的内存组合起来,成为一个“内存池”,不管是哪个服务器产生的 Sessoin 都可以放到这个“内存池”中,其他的 Web 服务器都可以使用。使用 Memcached 来同步 Session 的优点是:不会加大数据库的负担,并且安全性比 Cookie 高,把 Session 放到内存里面,读取速度比其他处理方式要快很多。

自定义使用 Memcached 处理 Session 信息,要使用到用户自定义会话存储函数 session_set_save_handler() ,自定义的 Session 类可以参考 ThinkPHP 3.2.3 full 版中的 Session 类(\ThinkPHP\Library\Think\Session\Driver\Memcache.class.php)

注:使用的 php 扩展为 memcache 1.4.24,服务器环境为 Windows7 + PHP 5.3.10 + Apache 2.2.21

memcached.class.php

<?php

class MemcacheSession {

    protected static $lifeTime     = 3600;
    protected static $handle       = null;

    public function start(Memcache $mem) {
        self::$handle = $mem;
        self::$lifeTime = ini_get(‘session.gc_maxlifetime‘);
        session_set_save_handler(
            array(__CLASS__, ‘open‘),
            array(__CLASS__, ‘close‘),
            array(__CLASS__, ‘read‘),
            array(__CLASS__, ‘write‘),
            array(__CLASS__, ‘destroy‘),
            array(__CLASS__, ‘gc‘)
        );
        session_start();
    }

    /**
     * 打开Session
     * @access private
     * @param string $savePath
     * @param mixed $sessName
     */
    private static function open($savePath, $sessName) {
        return true;
    }

    /**
     * 关闭Session
     * @access public
     */
    public static function close() {
        self::gc(self::$lifeTime);
        self::$handle->close();
        self::$handle = null;
        return true;
    }

    /**
     * 读取Session
     * @access private
     * @param string $sessID
     */
    private static function read($sessID) {
        return self::$handle->get($sessID);
    }

    /**
     * 写入Session
     * @access public
     * @param string $sessID
     * @param String $sessData
     */
    public static function write($sessID, $sessData) {
        return self::$handle->set($sessID, $sessData, 0, self::$lifeTime);
    }

    /**
     * 删除Session
     * @access private
     * @param string $sessID
     */
    private static function destroy($sessID) {
        return self::$handle->delete($sessID);
    }

    /**
     * Session 垃圾回收
     * @access private
     * @param string $sessMaxLifeTime
     */
    private static function gc($sessMaxLifeTime) {
        return true;
    }
}

使用 memcached.class.php

session.php

<?php
header(‘Content-type:text/html; charset=utf-8‘);

require ‘memcache.class.php‘;

$mem = new Memcache();
$mem->connect(‘127.0.0.1‘, 11211);

$memclass = new MemcacheSession();
$memclass->start($mem);

$_SESSION[‘username‘] = ‘dee‘;

echo ‘<a href="get.php" target="_blank">跳转</a>‘;

echo ‘<pre>‘;
print_r($_SESSION);
echo session_id(),‘<br />‘;

get.php

<?php

require ‘memcache.class.php‘;

$mem = new Memcache();
$mem->connect(‘127.0.0.1‘, 11211);

$memclass = new MemcacheSession();
$memclass->start($mem);

echo ‘<pre>‘;
print_r($_SESSION);
echo session_id();

二、Memcached 集群实现 Session 共享 

模拟开启两台 Memcached 服务器(127.0.0.1:11211 和 127.0.0.1:11212),引入 Memcached 的 Session 类, 同时修改 php.ini 中的配置

ini_set(‘memcache.hash_strategy‘, ‘consistent‘);//使用一致性hash算法
ini_set(‘session.save_path‘, ‘tcp://127.0.0.1:11211?weight=10,tcp://127.0.0.1:11212‘);

此时查看 phpinfo 的信息:

设置 Session

cluster_session_set.php

<?php
header(‘Content-type:text/html; charset=utf-8‘);

require ‘memcache.class.php‘;

ini_set(‘memcache.hash_strategy‘, ‘consistent‘);
ini_set(‘session.save_path‘, ‘tcp://127.0.0.1:11211?weight=10,tcp://127.0.0.1:11212‘);

$mem = new Memcache();
$mem->addServer("127.0.0.1",11211) or die ("Could not add server 11211");
$mem->addServer("127.0.0.1",11212) or die ("Could not add server 11212");  

$memclass = new MemcacheSession();
$memclass->start($mem);

$_SESSION[‘username‘] = "deathmask";  

echo session_id();
echo ‘<pre>‘;
print_r($_SESSION);
echo ‘<a href="cluster_session_get.php" target="_blank">跳转</a>‘;

输出:

打印 Session

cluster_session_get.php

<?php

require ‘memcache.class.php‘;

ini_set(‘memcache.hash_strategy‘, ‘consistent‘);
ini_set(‘session.save_path‘, ‘tcp://127.0.0.1:11211?weight=10,tcp://127.0.0.1:11212‘);

$mem = new Memcache();
$mem->addServer("127.0.0.1",11211) or die ("Could not add server 11211");
$mem->addServer("127.0.0.1",11212) or die ("Could not add server 11212");  

$memclass = new MemcacheSession();
$memclass->start($mem);

echo session_id(); 

echo ‘<pre>‘;
print_r($_SESSION);

输出:

使用 telnet 客户端分别连接 127.0.0.1:11211 和 127.0.0.1:11212,获得 session_id 所对应的值:

127.0.0.1:11211

127.0.0.1:11212

三、Memcached 存储 Session 的弊端

① Memcached 把内存分成很多种规格的存储块(chunk),这种方式决定了 Memcached 不能完全利用内存,会产生内存碎片,如果存储块不足,还会产生内存溢出;

② 当 Memcached 集群发生故障(比如内存溢出)或者维护(比如升级、增加或减少服务器)时,用户会无法登录,或者被踢掉线;

③ Memcached 的回收机制(LRU,Least Recently Used 近期最少使用算法)可能会导致用户无缘无故地掉线。

解决方案是可以使用 Memcached + MySQL:

a. 当用户登录时,将 Session “set”到 Memcached,并写入数据库;

b. 在 Session 中增加一个字段,标识 Session 最后写入数据库的时间;

c. 每个页面加载的时候,优先从 Memcached 读取 Session,其次从数据库读取;

d. 每加载 N 页或者 Y 分钟后,再次将 Session 写入数据库;

e. 从数据库中获取过期 Session,优先从 Memcached 中获取最新数据

或者使用 Redeis ,利用 Redis 的持久化来存储 Session。

参考:

集群中用Memcached来实现session共享

为什么不能用memcached存储Session

web集群时session同步的3种方法

如何严格设置php中session过期时间

利用memcached来做会话共享

Session变量不能传送到下一页.解决: session.use_trans_sid = 1

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

php模块memcache和memcached区别分析

关于memcache分布式一致性hash

memcached 内存分配(slab和chunk)

时间: 2024-10-05 23:23:18

Memcached 笔记与总结(9)Memcached 与 Session的相关文章

Memcached笔记——(三)Memcached使用总结

为了将N个前端数据同步,通过Memcached完成数据打通,但带来了一些新问题: 使用iBatis整合了Memcached,iBatis针对每台server生成了唯一标识,导致同一份数据sql会产生不同的key,造成重复缓存.——通过重写iBatis部分原码,终止了唯一标识的生成,同一个SQL产生同一个Key,同时对生成key做hash,控制长度,使得数据统一在Memcached. 为了迎合iBatis的架构,通过CacheModel模式,对缓存数据分组管理.最初通过Map实现CacheMode

Memcached笔记——(一)安装&amp;常规错误&amp;监控

08年的时候接触过Memcached,当时还对它的客户端产品嗤之以鼻,毕竟手工代码没有各种ORM原生XML配置方便.尽管如此,Memcached现在已经成了服务器架构里不可或缺的一部分! 相关链接: Memcached笔记——(一)安装&常规错误&监控Memcached笔记——(二)XMemcached&Spring集成 Memcached笔记——(三)Memcached使用总结 Memcached笔记——(四)应对高并发攻击  一.下载 1.Libevent 简单的说就是一个事件

Memcached笔记——(四)应对高并发攻击

近半个月过得很痛苦,主要是产品上线后,引来无数机器用户恶意攻击,不停的刷新产品各个服务入口,制造垃圾数据,消耗资源.他们的最好成绩,1秒钟可以并发6次,赶在Database入库前,Cache进行Missing Loading前,强占这其中十几毫秒的时间,进行恶意攻击. 相关链接: Memcached笔记——(一)安装&常规错误&监控Memcached笔记——(二)XMemcached&Spring集成 Memcached笔记——(三)Memcached使用总结  Memcached

Memcached笔记——(二)XMemcached&amp;Spring集成

今天研究Memcached的Java的Client,使用XMemcached 1.3.5,做个简单的测试,并介绍如何与Spring集成. 相关链接: Memcached笔记——(一)安装&常规错误&监控Memcached笔记——(二)XMemcached&Spring集成 Memcached笔记——(三)Memcached使用总结 Memcached笔记——(四)应对高并发攻击  一.Memcached Client简要介绍 Memcached Client目前有3种: Memca

Memcached笔记——(四)应对高并发攻击【转】

http://snowolf.iteye.com/blog/1677495 近半个月过得很痛苦,主要是产品上线后,引来无数机器用户恶意攻击,不停的刷新产品各个服务入口,制造垃圾数据,消耗资源.他们的最好成绩,1秒钟可以并发6次,赶在Database入库前,Cache进行Missing Loading前,强占这其中十几毫秒的时间,进行恶意攻击. 相关链接: Memcached笔记——(一)安装&常规错误&监控Memcached笔记——(二)XMemcached&Spring集成 Me

memcached 笔记之windows 7 下面 安装memcached 报错

windows 7 下面 安装memcached 报错 两种情况: 一:服务确实已经安装过 .如需要重新安装,当然是先memcached.exe -d uninstall 二:奇怪的是服务确实没有安装过的情况,在memcached.exe -d install也会出现上面定位报错.此时关掉cmd窗口  然后以管理员身份运行cmd,在执行上面的install,不出意外你会顺利的执行成功. 常用命令: 安装.卸载.启动.配置相关 -p 监听的端口 -l 连接的IP地址, 默认是本机 -d start

Key/Value之王Memcached初探:三、Memcached解决Session的分布式存储场景的应用

一.高可用的Session服务器场景简介 1.1 应用服务器的无状态特性 应用层服务器(这里一般指Web服务器)处理网站应用的业务逻辑,应用的一个最显著的特点是:应用的无状态性. PS:提到无状态特性,不得不说下Http协议.我们常常听到说,Http是一个无状态协议,同一个会话的连续两个请求互相不了解,他们由最新实例化的环境进行解析,除了应用本身可能已经存储在全局对象中的所有信息外,该环境不保存与会话有关的任何信息.之所以我们在使用ASP.NET WebForm开发中会感觉不到Http的无状态特

memcached笔记

启动memcached:./memcached -d -m 10 -l 127.0.0.1 -p 11211 -u root 连接memcached:telnet 127.0.0.1 11211 查看memcached所有key:stats items 结果: STAT items:16:number 1 STAT items:16:age 195 STAT items:16:evicted 0 STAT items:16:evicted_nonzero 0 STAT items:16:evic

Memcached 笔记与总结(7)增加虚拟节点

仅仅把 Memcached 服务器集群地址通过一致性哈希转映射在圆环上,可能会出现数据不能均匀地分配给各台 Memcached 服务器. 解决方案是引入虚拟节点,就是把每个映射在圆环上的服务器地址(物理节点)转变成更多的(注:关于虚拟节点的个数参考①)虚拟节点. 修改 Memcached 笔记与总结(6)PHP 实现 Memcached 的一致性哈希分布算法 中的代码: 类 consistentHash 增加私有的成员属性:$position,以键值形式保存所有虚拟节点的哈希值(键)和对应的服务

【转】 Key/Value之王Memcached初探:三、Memcached解决Session的分布式存储场景的应用

一.高可用的Session服务器场景简介 1.1 应用服务器的无状态特性 应用层服务器(这里一般指Web服务器)处理网站应用的业务逻辑,应用的一个最显著的特点是:应用的无状态性. PS:提到无状态特性,不得不说下Http协议.我们常常听到说,Http是一个无状态协议,同一个会话的连续两个请求互相不了解,他们由最新实例化的环境进行解析,除了应用本身可能已经存储在全局对象中的所有信息外,该环境不保存与会话有关的任何信息.之所以我们在使用ASP.NET WebForm开发中会感觉不到Http的无状态特