Home  >  Article  >  Backend Development  >  php 跨域、跨子域,跨服务器登录方法详解

php 跨域、跨子域,跨服务器登录方法详解

WBOY
WBOYOriginal
2016-06-20 13:04:491254browse


如何通过php跨域、跨子域或跨服务器读取session登录呢?本文为大家作详细介绍,有需要的朋友,可以参考下。

PHP跨子域和跨服务器

Session主要分两部分:

一个是Session数据,该数据默认情况下是存放在服务器的tmp文件下的,是以文件形式存在。

另一个是标志着Session数据的Session Id,Session ID,就是那个 Session 文件的文件名,Session ID 是随机生成的,因此能保证唯一性和随机性,确保 Session 的安全。一般如果没有设置 Session 的生存周期,则 Session ID 存储在内存中,关闭浏览器后该 ID 自动注销,重新请求该页面后,重新注册一个 session ID。如果客户端没有禁用 Cookie,则 Cookie 在启动 Session 会话的时候扮演的是存储 Session ID 和 Session 生存期的角色。

两个不同的域名网站,想用同一个Session,就是牵扯到Session跨域问题!
默认情况下,各个服务器会各自分别对同一个客户端产生 SESSIONID,如对于同一个用户浏览器,A 服务器产生的 SESSION ID 是 11111111111,而B 服务器生成的则是jbxuejbxue。另外,PHP 的 SESSION数据都是分别保存在本服务器的文件系统中。想要共享 SESSION 数据,那就必须实现两个目标:

一个是各个服务器对同一个客户端产生的SESSION ID 必须相同,并且可通过同一个 COOKIE 进行传递,也就是说各个服务器必须可以读取同一个名为 PHPSESSID 的COOKIE;另一个是 SESSION 数据的存储方式/位置必须保证各个服务器都能够访问到。这两个目标简单地说就是多服务器(A、B服务器)共享客户端的 SESSION ID,同时还必须共享服务器端的 SESSION 数据。

第一个目标的实现其实很简单,只需要对 COOKIE 的域(domain)进行特殊地设置即可(setcookie()函数中的第4个参数),默认情况下,COOKIE 的域是当前服务器的域名/IP 地址,而域不同的话,各个服务器所设置的 COOKIE 是不能相互访问的。

1)跨子域
采用这种方式,跨域不行,但同一子域可以,如:aaa.jbxue.com 和www.jbxue.com 都属于域 .jbxue.com是可以的,那么我们就可以设置 COOKIE 的域为 .jbxue.com,这样 aaa.jbxue.com、www.jbxue.com等等都可以访问此COOKIE。这样各个服务器共享同一客户端 SESSION ID 的目的就达到了。

1、只要在php页面的最开始(要在任何输出之前,并且在session_start()之前)的地方进行以下设置
 

ini_set('session.cookie_path', '/');
ini_set('session.cookie_domain', '.mydomain.com');
ini_set('session.cookie_lifetime', '1800');

2、在php.ini里设置
 

session.cookie_path = /
session.cookie_domain = .mydomain.com

session.cookie_lifetime = 1800

3、在php页面最开始的地方(条件同1)调用函数

session_set_cookie_params(1800 , '/', '.mydomain.com');


这三种方式都是同样的效果。

这里我用第一种方法设置,分别在www.mydomain.com和sub.mydomain.com两个域名来测试。
sub1.php

sub2.php
 

访问顺序:
(1)www.mydomain.com/sub1.php
页面输出:Array ( [sub1] => sub1 )

(2)sub.mydomain.com/sub2.php
页面输出:Array ( [sub1] => sub1 [sub2] => sub2 )

成功

第二个目标的实现可以使用数据库来保存SESSION 数据,这样各个服务器就可以方便地访问同一个数据源,获取相同的SESSION 数据了;或者是通过文件共享方式,如 NFS 方式(我的其他文章有如何配置nfs)
如果用数据库存储session数据的话,可能会有遗留问题,就是如果网站的访问量很大的话,SESSION 的读写会频繁地对数据库进行操作,可以把这个放在memcache中。存放在数据库里的前面有文章实现了。把数据库和memcache结合的思路,前面有了。如果单独用memcache存放session不太好,最好和数据库结合操作。

2)跨域解决
思路:

用jsonp跨域方法获取另外一个域的登录加密信息,实现本域登录。
或者用iframe解决,但是ff/IE不支持,所以需要前面加上p3p协议。

P3P(Platform for Privacy Preferences Project),是个协议,通过其声明它是好人,允许收集浏览器用户行为。
可现实中,大家都可以说自己是好人,背地里没准儿干啥坏事呢。
这就是其分歧所在。[参考] 国内多数网站,都不关注这个 P3P。隐私问题可能没国外(微软的隐私声明)重视吧。

首先想到就是通过JS操作Cookie并让两个不同域的cookie能够相互访问,这样就可达到了上述的效果。

以下是具体实现过程,分两个步骤:
1、在A系统下成功登录后,利用JS动态创建一个隐藏的iframe,通过iframe的src属性将A域下的cookie值作为get参数重定向到B系统下b.jsp页面上;
 

    var _frm = document.createElement("iframe");  
    _frm.style.display="none";  
    _frm.src = "http://www.jbxue.com/setcookie.php?mycookie=xxxxx";//此处xxx最好编码  
    document.body.appendChild(_frm); 

2、在B系统的setcookie.php页面中来获取A系统中所传过来的cookie值,并将所获取到值写入用户的cookie中,当然域是自己的了,这样就简单的实现了cookie跨域的访问; 不过这其中有个问题需要注意,就是在IE浏览器下这样操作不能成功,需要在setocokie.php页面中设置P3P HTTP Header就可以解决了(具体詳細信息可以参考:http://www.w3.org/P3P/),P3P设置代码为:
 

header('P3P: CP="CURa ADMa DEVa PSAo PSDo OUR BUS UNI PUR INT DEM STA PRE COM NAV OTC NOI DSP COR"');//ecshop这么设置的

 


Statement:
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn