美文网首页程序员让前端飞
JS跨域问题解决方案

JS跨域问题解决方案

作者: 半亩房顶 | 来源:发表于2019-03-05 16:32 被阅读32次

    一、JSONP

    在js中,我们直接用XMLHttpRequest请求不同域上的数据时,是不可以的。但是,在页面上引入不同域上的js脚本文件却是可以的,script标签里的src属性来完成的,jsonp正是利用这个特性来实现的。
    比如,在桌面新建一个crossDomain.html页面,它里面的代码需要利用ajax获取一个不同域上的json数据,假设这个json数据地址是http://192.168.x.xxx/JSONP/jsonpTest.php那么crossDomain.html中的代码就可以这样:

    <script type="text/javascript">
    var text = document.querySelector('.text');
    function dosomething(jsondata) {
        var str = "";
        for (var i = 0; i < jsondata.length; i++) {
            str += jsondata[i];
        }
        text.innerHTML = '我是JS通过JSONP跨域请求来的数据:'+'<span class="show">'+str+'</span>';
    }
    </script>
    <script type="text/javascript" src="http://192.168.x.xxx/JSONP/jsonpTest.php?callback=dosomething"></script>
    

    可以看到在获取数据的地址后面还有一个callback参数,按惯例是用这个参数名,但是你用其他的也一样。当然如果获取数据的jsonp地址页面不是你自己能控制的,就得按照提供数据的那一方的规定格式来操作了。

    因为是当做一个js文件来引入的,所以http://192.168.x.xxx/JSONP/jsonpTest.php返回的必须是一个能执行的js文件,所以这个页面的php代码可能是这样的:

    <?php
        $callback = $_GET['callback'];//得到回掉函数名
        $data = array('a','b','c'); //要返回的数据
        echo $callback.'('.json_encode($data).')'; //输出
    ?>
    

    然后在crossDomain.html中打印出返回的jsondata如下:

    ["a", "b", "c"]

    可以看到请求成功了,然后就可以在crossDomain.html这个页面里处理这个数据了。
    这样jsonp的原理就很清楚了,通过script标签引入一个js文件,这个js文件载入成功后会执行我们在url参数中指定的函数,并且会把我们需要的json数据作为参数传入。所以jsonp是需要服务器端的页面进行相应的配合的。
    当然可以直接用一些已经封装过的库,这样就不用每次去创建script标签了。如下为JQ的跨域API:

        $.getJSON('http://192.168.x.xxx/JSONP/jsonpTest.php?callback=?',function(jsondata){
            console.log(jsondata);//["a", "b", "c"]
            var str = "";
            $.each(jsondata,function(i,index){
                return str += index;
            });
            $(".text1").html('我是JQ通过JSONP跨域请求来的数据:'+'<span class="show">'+str+'</span>');
        });
    

    jquery的getJSON方法会自动生成一个全局函数来替换callback=?中的问号,之后获取到数据后又会自动销毁,实际上就是起一个临时代理函数的作用。$.getJSON方法会自动判断是否跨域,不跨域的话,就调用普通的ajax方法;跨域的话,则会以异步加载js文件的形式来调用jsonp的回调函数。


    二、CORS

    CORS是跨源资源分享(Cross-Origin Resource Sharing)的缩写。它是W3C标准,是跨源AJAX请求的根本解决方法。相比JSONP只能发GET请求,CORS允许任何类型的请求。CORS需要浏览器和服务器同时支持。目前,所有浏览器都支持该功能,IE浏览器不能低于IE10。
    整个CORS通信过程,都是浏览器自动完成,不需要用户参与。对于开发者来说,CORS通信与同源的AJAX通信没有差别,代码完全一样。浏览器一旦发现AJAX请求跨源,就会自动添加一些附加的头信息,有时还会多出一次附加的请求,但用户不会有感觉。
    因此,实现CORS通信的关键是服务器。只要服务器实现了CORS接口,就可以跨源通信。

    服务端设置Access-Control-Allow-Origin
    这种方式只要服务端把response的header头中设置Access-Control-Allow-Origin为请求当前域名下数据的域名即可。一般情况下设为 * 即可。这样客户端就不需要使用jsonp来获取数据。

    会有安全问题(更多请百度)

    但是个人是用这个方法的

    CORS支持POST提交,并且实施起来很简单,CORS原理:只需要向响应头header中注入Access-Control-Allow-Origin,这样浏览器检测到header中的Access-Control-Allow-Origin,则就可以跨域操作。

    我用的是php,用法如:(*号也可以指定特定的域名,只允许该域名访问)

     <?php    
        header("Access-Control-Allow-Origin:*");     
        //...
    

    三、通过修改document.domain来跨子域

    上面的jsonp是来解决ajax跨域请求的,那么如果是需要处理 Cookie 和 iframe 该怎么办呢?这时候就可以通过修改document.domain来跨子域。两个网页一级域名相同,只是二级域名不同,浏览器允许通过设置document.domain共享 Cookie或者处理iframe。比如A网页是http://w1.example.com/a.html,B网页是http://w2.example.com/b.html,那么只要设置相同的document.domain,两个网页就可以共享Cookie。

    document.domain = 'example.com';
    //现在,A网页通过脚本设置一个 Cookie。
    document.cookie = "test1=hello";
    //B网页就可以读到这个 Cookie。
    var allCookie = document.cookie;
    

    注意,这种方法只适用于 Cookie 和 iframe 窗口,LocalStorage 和 IndexDB 无法通过这种方法,规避同源政策,而要使用下文介绍的PostMessage API。
    另外,服务器也可以在设置Cookie的时候,指定Cookie的所属域名为一级域名,比如.example.com。

    Set-Cookie: key=value; domain=.example.com; path=/
    //这样的话,二级域名和三级域名不用做任何设置,都可以读取这个Cookie。
    

    不同的iframe 之间(父子或同辈),是能够获取到彼此的window对象的,但是你却不能使用获取到的window对象的属性和方法(html5中的postMessage方法是一个例外,还有些浏览器比如ie6也可以使用top、parent等少数几个属性),总之,你可以当做是只能获取到一个几乎无用的window对象。
    首先说明一下同域之间的iframe是可以操作的。比如http://127.0.0.1/JSONP/a.html里面嵌入一个iframe指向http://127.0.0.1/myPHP/b.html。那么在a.html里面是可以操作iframe里面的DOM的。

    <iframe src="http://127.0.0.1/myPHP/b.html" frameborder="1"></iframe>
    <body>
    <script type="text/javascript">
    var iframe = document.querySelector("iframe");
    iframe.onload = function(){
        var win = iframe.contentWindow;
        var doc = win.document;
        var ele = doc.querySelector(".text1");
        var text = ele.innerHTML="123456";
    }
    </script>
    

    如果两个网页不同源,就无法拿到对方的DOM。典型的例子是iframe窗口和window.open方法打开的窗口,它们与父窗口无法通信。如果两个窗口一级域名相同,只是二级域名不同,那么document.domain属性,就可以规避同源政策,拿到DOM。
    对于完全不同源的网站,目前有三种方法,可以解决跨域窗口的通信问题。

    片段识别符(fragment identifier)
    window.name
    跨文档通信API(Cross-document messaging)

    相关文章

      网友评论

        本文标题:JS跨域问题解决方案

        本文链接:https://www.haomeiwen.com/subject/nqdouqtx.html