zoukankan      html  css  js  c++  java
  • 采用EaglePHP框架解决分布式集群服务器利用MEMCACHE方式共享SESSION数据的问题

    一、问题起源

    稍大一些的网站,通常都会有好几个服务器,每个服务器运行着不同功能的模块,使用不同的二级域名,而一个整体性强的网 站,用户系统是统一的,即一套用户名、密码在整个网站的各个模块中都是可以登录使用的。各个服务器共享用户数据是比较容易实现的,只需要在后端放个数据库 服务器,各个服务器通过统一接口对用户数据进行访问即可。但还存在一个问题,就是用户在这个服务器登录之后,进入另一个服务器的别的模块时,仍然需要重新 登录,这就是一次登录,全部通行的问题,映射到技术上,其实就是各个服务器之间如何实现共享 SESSION 数据的问题。

    二、PHP SESSION 的工作原理

    在 解决问题之前,先来了解一下 PHP SESSION 的工作原理。在客户端(如浏览器)登录网站时,被访问的 PHP 页面可以使用 session_start() 打开 SESSION,这样就会产生客户端的唯一标识 SESSION ID(此 ID 可通过函数 session_id() 获取/设置)。SESSION ID 可以通过两种方式保留在客户端,使得请求不同的页面时,PHP程序可以获知客户端 的 SESSION ID;一种是将 SESSION ID 自动加入到 GET 的 URL 中,或者 POST的表单中,默认情况下,变量名 为 PHPSESSID;另一种是通过 COOKIE,将 SESSION ID 保存在 COOKIE中,默认情况下,这个 COOKIE 的名字 为 PHPSESSID。这里我们主要以 COOKIE方式进行说明,因为应用比较广泛。

    那么 SESSION 的数据保存在哪里呢?当 然是在服务器端,但不是保存在内存中,而是保存在文件或数据库中。默认情况下,php.ini中设置的 SESSION 保存方式 是 files(session.save_handler = files),即使用读写文件的方式保存SESSION 数据,而 SESSION 文 件保存的目录由 session.save_path 指定,文件名以 sess_为前缀,后跟 SESSION ID, 如:sess_c72665af28a8b14c0fe11afe

    3b59b51b。文件中的数据即是序列化之后 的 SESSION 数据了。如果访问量大,可能产生的 SESSION文件会比较多,这时可以设置分级目录进行 SESSION文件的保存,效率会提高 很多,设置方法为:session.save_path=”N;/save_path”,N为分级的级数,save_path 为开始目录。当写 入 SESSION 数据的时候,PHP 会获取到客户端的SESSION_ID,然后根据这个 SESSION ID 到指定的 SESSION 文件 保存目录中找到相应的 SESSION文件,不存在则创建之,最后将数据序列化之后写入文件。读取 SESSION数据是也是类似的操作流程,对读出来的 数据需要进行解序列化,生成相应的 SESSION 变量。 

    三、多服务器共享 SESSION 的主要障碍及解决办法 

    通 过了解 SESSION 的工作原理,我们可以发现,在默认情况下,各个服务器会各自分别对同一个客户端产生 SESSIONID,如对于同一个用户浏览 器,A 服务器产生的 SESSION ID 是30de1e9de3192ba6ce2992d27a1b6a0a,而 B 服务器生成的则是 c72665af28a8b14c0fe11afe3b59b51b。另外,PHP 的 SESSION数据都是分别保存在本服务器的文件系统中。 

    确 定了问题所在之后,就可以着手进行解决了。想要共享 SESSION 数据,那就必须实现两个目标:一个是各个服务器对同一个客户端产生的 SESSION ID 必须相同,并且可通过同一个 COOKIE 进行传递,也就是说各个服务器必须可以读取同一个名为 PHPSESSID 的 COOKIE;另一个是 SESSION 数据的存储方式/位置必须保证各个服务器都能够访问到。简单地说就是多服务器共享客户端的 SESSION ID,同时还必须共享服务器端的 SESSION 数据。 

    第一个目标的实现其实很简单,只需要对 COOKIE 的域 (domain)进行特殊地设置即可,默认情况下,COOKIE的域是当前服务器的域名/IP 地址,而域不同的话,各个服务器所设置 的 COOKIE 是不能相互访问的,如 www.aaa.com的服务器是不能读写 www.bbb.com 服务器设置的 COOKIE的。这里我们 所说的同一网站的服务器有其特殊性,那就是他们同属于同一个一级域,如:demo.eaglephp.com 和www.eaglephp.com 都属 于域 .eaglephp.com,那么我们就可以设置 COOKIE 的域为.eaglephp.com,这样 demo.eaglephp.com、 www.eaglephp.com 等等都可以访问此COOKIE。PHP 代码中的设置方法如下:

    <?php

    Session::cookieDomain('.eaglephp.com');// 基于EaglePHP框架开发

    ?> 
    这样各个服务器共享同一客户端 SESSION ID 的目的就达到了。 

    第二个目标的实现可以使用文件共享方式,有2种方式可以解决,一是用数据库存session,还有就是试用memcache。这里用MEMCACHE来解决. 

    我 用的是EaglePHP框架,已经支持memcache方式存取session.架好memcache服务器后,只需要在配置文件里面设置好 memcache的IP和端口,然后指定COOKIE_DOMAIN参数就可以了,然后就可以按正常操作session的方式进行操作,这时已经可以多域 名共享session了。

    在此处说明下,在EaglePHP框架采用memcache保存session的方式:

    修改EaglePHP/Lib/Main.inc.php中的SESSION_SAVE_TYPE常量,设置为memcache,另外在自带的cms后台参数设置处修改memcache的host和ip。

    http://www.oschina.net/question/196828_69013

  • 相关阅读:
    Haskell Interactive Development in Emacs
    Access Java API in Groovy Script
    手工设置Eclipse文本编辑器的配色
    Color Theme of Emacs
    Gnucash的投资记录
    Special Forms and Syntax Sugars in Clojure
    Use w3m as Web Browser
    SSE指令集加速之 I420转BGR24
    【图像处理】 增加程序速度的方法
    TBB 入门笔记
  • 原文地址:https://www.cnblogs.com/hellowzd/p/5639988.html
Copyright © 2011-2022 走看看