1.在解决session跨域问题前,首先先了解下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_c72665af28a8b14c0fe11afe3b5Arrayb51b。文件中的数据即是序列化之后的 SESSION 数据了。如果访问量大,可能产生的 SESSION 文件会比较多,这时可以设置分级目录进行 SESSION 文件的保存,效率会提高很多,设置方法为:session.save_path=”N;/save_path”,N 为分级的级数,save_path 为开始目录。当写入 SESSION 数据的时候,PHP 会获取到客户端的 SESSION_ID,然后根据这个 SESSION ID 到指定的 SESSION 文件保存目录中找到相应的 SESSION 文件,不存在则创建之,最后将数据序列化之后写入文件。读取 SESSION 数据是也是类似的操作流程,对读出来的数据需要进行解序列化,生成相应的 SESSION 变量。
2.多服务器共享SESSION 的主要障碍及解决办法
通过了解 SESSION 的工作原理,我们可以发现,在默认情况下,各个服务器会各自分别对同
一个客户端产生 SESSION ID,如对于同一个用户浏览器,A 服务器产生的 SESSION ID 是 30de1eArrayde31Array2ba6ce2ArrayArray2d27a1b6a0a,而 B 服务器生成的则是 c72665af28a8b14c0fe11afe3b5Arrayb51b。另外,PHP 的 SESSION 数据都是分别保存在
本服务器的文件系统中。
确定了问题所在之后,就可以着手进行解决了。想要共享 SESSION 数据,那就必须实现两个目标:一个是各个服务器对同一个客户端产生的 SESSION ID 必须相同,并且可通过同一个 COOKIE 进行传递,也就是说各个服务器必须可以读取同一个名为 PHPSESSID 的 COOKIE;
另一个是 SESSION 数据的存储方式/位置必须保证各个服务器都能够访问到。 简单地说就是多
服务器共享客户端的 SESSION ID,同时还必须共享服务器端的 SESSION 数据。
第一个目标的实现其实很简单,只需要对 COOKIE 的域(domain)进行特殊地设置即可,
默认情况下,COOKIE 的域是当前服务器的域名/IP 地址,而域不同的话,各个服务器所设
置的 COOKIE 是不能相互访问的。
3.解决问题
贴上代码先
//加载session,用于设置其修改session的域名网站
ini_set('session.cookie_domain', ".test.cn");//跨域访问Session
//设置名称,该名称是任意改的,主要是用于session保存在本地cookie的名称
ini_set('session.name', "session_name");
//设置过期时间
ini_set('session.gc_maxlifetime', "3600");
//打开sesseion
session_start();
//如果cookie存在,则证明session也存在,把session key读取出来
if(isset($_COOKIE["session_name")
]))
{
//存在cookie,读取sessionID
session_id($_COOKIE["session_name")
]);
}
else
{
//不存在cookie,证明session也不存在,重新赋值一个session id 给cookie
$_COOKIE[C("SESSION_NAME")]=session_id();
}
总上所述,容易看得出,session的获取,其实是根据一个保存在服务器端的session文件读取的,session文件的读取则根据sessionid读取的,只要在使用session前,先设置sessionid,程序则会自动找回那个session文件,如果文件里面有值,则证明用户在其别的网站以使用过改session,从而,我们就可以直接读取该session的值了,这个是最简单的共享,还没涉及到加密和验证,有需要,则自行添加修改。