浏览器跨域问题

一、什么是跨域?
跨域是指从一个域名的网页去请求另一个域名的资源。比如从www.baidu.com 页面去请求 www.google.com 的资源。跨域的严格一点的定义是:只要 协议,域名,端口有任何一个的不同,就被当作是跨域,由于浏览器的同源策略造成的。
二、浏览器为什么要限制跨域访问呢
1.浏览器的同源策略:何为同源:域名,协议,端口号均相同。
2.同源策略分为两种:
  1. DOM同源策略:禁止对不同源页面DOM进行操作。这里主要场景是iframe跨域的情况,不同域名的iframe是限制互相访问的。
  2. XmlHttpRequest同源策略:禁止使用XHR对象向不同源的服务器地址发起HTTP请求
  3. AJAX同源策略主要用来防止CSRF攻击。如果没有AJAX同源策略,相当危险,我们发起的每一次HTTP请求都会带上请求地址对应的cookie,那么可以做如下攻击.
  1. 用户登录了自己的银行页面 mybank.commybank.com向用户的cookie中添加用户标识。
  2. 用户浏览了恶意页面 evil.com。执行了页面中的恶意AJAX请求代码。
  3. evil.commybank.com发起AJAX HTTP请求,请求会默认把mybank.com对应cookie也同时发送过去。
  4. 银行页面从发送的cookie中提取用户标识,验证用户无误,response中返回请求数据。此时数据就泄露了。
  5. 而且由于Ajax在后台执行,用户无法感知这一过程。
  6. DOM同源策略也一样,如果iframe之间可以跨域访问,可以这样攻击:
    1. 做一个假网站,里面用iframe嵌套一个银行网站 mybank.com
    2. 把iframe宽高啥的调整到页面全部,这样用户进来除了域名,别的部分和银行的网站没有任何差别。
    3. 这时如果用户输入账号密码,我们的主网站可以跨域访问到mybank.com的dom节点,就可以拿到用户的输入了,那么就完成了一次攻击。

    所以说有了跨域跨域限制之后,我们才能更安全的上网了


三、跨域访问的解决方案
       1.服务器端响应头设置:
 
   
  1. header("Access-Control-Allow-Origin: *");
 
    
  1. header("Access-Control-Allow-Origin: yourhost.com");
 
     
  1. header('Access-Control-Allow-Method:POST,GET');//允许访问的方式

2.jsonp实现跨域
        原理: 通过动态创建script标签,然后利用src属性进行跨域。浏览器对script,img的src属性是不做跨域限制的。
列子:simple.html
 
   
  1. <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
  2. <html xmlns="http://www.w3.org/1999/xhtml" >
  3. <head>
  4. <title>test</title>
  5. <script type="text/javascript">
  6. function addScriptTag(src){
  7. var script = document.createElement('script');
  8. script.setAttribute("type","text/javascript");
  9. script.src = src;
  10. document.body.appendChild(script);
  11. }
  12. window.onload = function(){
  13. //搜索apple,将自定义的回调函数名result传入callback参数中
  14. addScriptTag('http://127.0.0.1/kuayu/demo.php?v=1.0&q=apple&callback=result');
  15. };
  16. //自定义的回调函数result
  17. function result(data) {
  18. //我们就简单的获取apple搜索结果的第一条记录中url数据
  19. alert(data.Name);
  20. }
  21. </script>
  22. </head>
  23. <body>
  24. </body>
  25. </html>
服务器端:
 
    
  1. <?php
  2. #header("Connection: Close");
  3. #header("http/1.1 404 Not Found test");
  4. #header("Access-Control-Allow-Origin: *");
  5. #header("Access-Control-Allow-Origin: yourhost.com");
  6. $arr = array('Name'=>'jian','age'=>'26');
  7. $callback = $_GET['callback'];
  8. $json = json_encode($arr);
  9. echo "$callback($json)";
  10. #echo json_encode($arr);
  11. ?>
Jquery对jsonp的实现
列子:getJson.html
 
    
  1. <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
  2. <html xmlns="http://www.w3.org/1999/xhtml" >
  3. <head>
  4. <title>test</title>
  5. <script src="http://libs.baidu.com/jquery/2.0.0/jquery.min.js"></script>
  6. <script type="text/javascript">
  7. var url="http://127.0.0.1/kuayu/demo.php?callback=?";
  8. $(function(){
  9. $.getJSON(url,function(data){
  10. alert (data.Name);
  11. })
  12. });
  13. </script>
  14. </head>
  15. <body>
  16. </body>
  17. </html>
服务端:
 
    
  1. <?php
  2. #header("Connection: Close");
  3. #header("http/1.1 404 Not Found test");
  4. #header("Access-Control-Allow-Origin: *");
  5. #header("Access-Control-Allow-Origin: yourhost.com");
  6. $arr = array('Name'=>'jian','age'=>'26');
  7. $callback = $_GET['callback'];
  8. $json = json_encode($arr);
  9. echo "$callback($json)";
  10. #echo json_encode($arr);
  11. ?>
注意点:
a.发送到数据接收方的地址后面一定要加上callback=?这样的参数,且这个?是会被Jquery自动替换成回调方法的名称。(在Jquery1.4中可以自己指定回调方法的名称了
b.由于调用Jquery的$.getJSON方法时,Jquery有自己的处理,实际上通过script的scr请求的,但要知道,数据最终还是通过url后面通过get方式发送数据出去的,这就决定了,发送的data数据量不能太多,否则造成url太长接收失败(getJSON方式是不可能有post方式递交的)。
3.代理
例如www.a.com/index.html需要调用www.b.com/index.php,可以写一个接口www.a.com/index.php,由这个接口在后端去调用 www.b.com/index.php并拿到返回值,然后再返回给index.html,这就是一个代理的模式。相当于绕过了浏览器端,自然就不存在跨域问题。

猜你喜欢

转载自blog.csdn.net/angle_jian/article/details/79491142