先来句题外话,最开始Ajax应该是用来特指用XMLHttpRequest传输数据这门技术,但就像最近大家把一切web新技术都归到html5名下一样,现在一切异步获取数据的手段都被人称之为Ajax。
由于JavaScript同源策略的存在,跨域数据交互是个老生常谈的话题了。网上相关文章很多,不过随着时间的推移和浏览器的更新,一部分解决方案已经不适用了,同时也出现了一些更好的方法。抛开纯服务器Proxy这种跟前端没什么关系的方案不说,这里简单总结下常见的其他几种方式。
JSONP
JSONP是最常见的跨域数据交互的方式,原理是html的script标签可以加载并执行其他域JS文件。站点B把要提供的数据作为参数传给一个站点A定义的全局函数,站点A引用这个文件就可以跨域获取数据了,A站还可以把少量参数放在script标签的src里提交给B站。外链JS这种方案只支持GET,受IE下url长度不能超过2083个字节的限制和出于安全考虑,一般不用来提交数据。
有人通过后端Proxy使得这种方式可以获取任意页面内容,还增加了对POST的支持,如:
HTML<script type="text/javascript" src="http://www.ajax-cross-domain.com/cgi-bin/ACD/ACD.js?uri=(http://www.google.com)"></script>
<script type="text/javascript">
alert(ACD.responseText);
</script>
<script type="text/javascript" src="http://www.ajax-cross-domain.com/cgi-bin/ACD/ACD.js?uri=(http://216.92.131.147/dotserv/ACD/runit/post.cgi)&method=post&postdata=(name=fred&[email protected])"></script>
<script type="text/javascript">
alert(ACD.responseText);
</script>
实际上这个方案是借助后端把任意页面输出为JS变量,后端根据url中相关标识来决定请求方式和参数,并不能解决大数据提交问题。
原生表单+Redirect+Callback
原生的form表单支持提交数据到其他域,我们只需要把form的target指向页面上的隐藏iframe,那就实现了无刷新提交,剩下的问题就是怎么获取提交后的结果。例如站点A表单提交数据到站点B,通常我们会在站点B处理完请求,重定向到站点A下某个Proxy页面,并在url带上参数标识处理结果。接着,A站下的Proxy页面就可以解析url参数,传给父页面的Callback函数来处理了。
Flash
利用flash的URLLoader,也可以轻松实现跨域数据交互。只要站点B的跨域策略文件(crossdomain.xml)中包含了站点A,A站就可以获取B站的数据,提交数据给B站。我们可以把JS和flash的交互封装一下,更方便的使用。这里有一个别人封装好的版本,使用起来和原生的XMLHttpRequest几乎一模一样:
JSvar req;
function callback() {
if (req.readyState == 4) {
try {
if (req.status != 200) {
alert(‘error detected 1‘);
} else {
alert("got data: "+req.responseText);
}
} catch(e) {
alert(‘error detected 2‘);
}
}
}
function test_get() {
req = new CrossXHR();
req.onreadystatechange = callback;
req.open(‘GET‘, ‘http://www.pliantdev.com/support/test.xml‘);
req.send();
}
Iframe+XMLHttpRequest
如果站点B有一个proxy页面,用原生Ajax(XMLHttpRequest)对B站其他页面进行各种数据交互,那么我们在A站用iFrame引入这个proxy页面,只需要解决iFrame跨域问题就可以了。实际上,如果A和B属于相同大域,设置两边的document.domain为根域名就OK了;如果是完全不同的两个域,也有许多现成的解决方案,例如经典的window.name。更妙的是,除开IE6、IE7,几乎所有现代浏览器都支持用window.postMessage实现不同iFrame的数据通讯。pmxdr就是这样一个库,利用postMessage把数据传给隐藏的站外iFrame来实现跨域Ajax,libxdr对它进行了进一步的封装,使之更好用:
JSvar req = new XDR();
req.open("POST", "http://code.eligrey.com/pmxdr/libxdr/demo.php");
req.setRequestHeader("Content-Type",
"application/x-www-form-urlencoded");
req.onload = function() {
alert(this.responseText); // alerts "foo is bar"
};
request.send("foo=bar");
值得注意的是,这种方案只需要在站点B部署一个proxy页面,其他任意站点都可以通过这个页面与之交互,不太安全,这一点pmxdr考虑到了,在pmxdr-host.js里有一个变量alwaysTrustedOrigins,它是一个数组,支持用正则定义允许交互的站点。
终极解决方案:CORS
实际上,除了IE6、IE7,大部分现代浏览器已经支持了跨域资源共享(Cross-Origin Resource Sharing,简称:CORS)标准,这可谓是跨域Ajax的终极解决方案。有了这个标准,只需要在Response Header里加上这么一条,就可以轻松跨域了:
Access-Control-Allow-Origin: http://hello-world.example
这个header定义允许哪些域跟自己交互,如果定义为*就表示允许任何域,这么做当然是不推荐的。在除IE之外的标准浏览器,这样就可以跨域Ajax了。对于IE,需要换用新增的XDomainRequest对象来发送请求,其它都类似。另外还有几个header可以用来设置允许的提交方式等信息,如果要支持认证或者提交xml等格式的数据给服务器,则需要预请求,这里有更多说明。
总结
不同的方案有各自不同的使用场景,谁好谁坏不能一概而论。一般的,跨域获取数据个人习惯用JSONP,跨域提交数据个人习惯用表单+Callback。随着现代浏览器的普及,原生xhr也可以尝试下,flash可以作为替补降级用。最后,如果要选用一个封装了多种跨域实现的库,可以考虑下Yui3的io组件。
本文提到一些组件链接:
- AJAX Cross Domain(在服务端把任意网页转为JS变量)
- Cross Domain XHR(基于flash,接口与xhr高度一致)
- pmxdr(使用postMessage和iframe的跨域方案)
- libxdr(pmxdr的再次封装,接口模拟IE的XDomainRequest)
- Yui3的IO组件(基于flash或原生xhr的跨域方案)