JSONP_跨域

2017-07-20  本文已影响0人  山门龙龙

1.什么是同源策略

浏览器出于安全考虑,只能同本域进行接口交互。不同源的客户端脚本在未经许可的情况下,不允许读写对方的资源。
本域指的是协议、域名、端口都要相同:
1.协议,比如http和https是不同协议
2.域名,比如https://jirengu.com/ahttps://jirengu.com/b是不同域名
3.端口,比如https://jirengu.com:80https://jirengu.com:8080是不同端口

2. 什么是跨域?跨域有几种实现形式

3.JSONP的原理是什么

4.CORS是什么

全称cross origin resource sharing,译作跨域资源共享。发送一个跨域请求,前端不做任何处理,而是在后端加一个Access-Control-Allow-Origin,地址设为前端的url,这样只要是前端url发送过来的请求,都可以反问后端的接口。如果设为" * ",则所有的url都可以访问。
CORS 分为简单请求和复杂请求 , 具体可以查看阮一峰的BLOG

5.根据视频里的讲解演示三种以上跨域的解决方式 ,写成博客

1.JSONP
前端代码如下:

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
    <title>news</title>
    <style>
        .container{
            width: 900px;
            margin: 0 auto;
        }
    </style>
</head>
<body>
<div class="container">
    <ul class="news">
        <li>第11日前瞻:中国冲击4金 博尔特再战</li>
        <li>男双力争会师决赛 </li>
        <li>女排将死磕巴西!</li>
    </ul>
    <button class="change">换一组</button>
</div>

<script>
    var btn = document.querySelector('button');
    btn.addEventListener('click',function () {
        var script = document.createElement('script');
        script.src = 'http://localhost:8080/getNews?callback=appendHtml';
        document.head.appendChild(script);
        document.head.removeChild(script);
    })
    function appendHtml(news) {
        var lis = '';
        for(var i=0; i<news.length; i++){
            lis += '<li>' + news[i] + '</li>';
        }
        console.log(lis);
        var ul = document.querySelector('ul');
        ul.innerHTML = lis;
    }
</script>
</body>
</html>

后端代码如下:

router.get('/getNews',function (req,res) {
    var news = [
        "第11日前瞻:中国冲击4金 博尔特再战200米羽球",
        "正直播柴飚/洪炜出战 男双力争会师决赛",
        "女排将死磕巴西!郎平安排男陪练模仿对方核心",
        "没有中国选手和巨星的110米栏 我们还看吗?",
        "中英上演奥运金牌大战",
        "博彩赔率挺中国夺回第二纽约时报:中国因对手服禁药而丢失的奖牌最多",
        "最“出柜”奥运?同性之爱闪耀里约",
        "下跪拜谢与洪荒之力一样 都是真情流露"
    ];
    var data = []
    for(var i=0; i<3 ;i++){
        var index = parseInt(Math.random() * news.length);
        data.push(news[index]);
        news.splice(index,1);
    }
    var callback = req.query.callback;
    if(callback){
        res.send(callback + '(' + JSON.stringify(data) + ')');
    }else{
        res.send(data);
    }
})

2.CORS
前端代码如下:

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
    <title>news</title>
    <style>
        .container{
            width: 900px;
            margin: 0 auto;
        }
    </style>
</head>
<body>
<div class="container">
    <ul class="news">
        <li>第11日前瞻:中国冲击4金 博尔特再战</li>
        <li>男双力争会师决赛 </li>
        <li>女排将死磕巴西!</li>
    </ul>
    <button class="change">换一组</button>
</div>

<script>
    var btn = document.querySelector('button');
    btn.addEventListener('click', function() {
        var xhr = new XMLHttpRequest();
        xhr.open('get', 'http://localhost:8080/getNews', true);
        xhr.send();
        xhr.onreadystatechange = function () {
            if (xhr.readyState === 4 && xhr.status === 200) {
                appendHtml(JSON.parse(xhr.responseText));
            }
        }
    });

    function appendHtml(news) {
     var lis = '';
     for(var i=0; i<news.length; i++){
         lis += '<li>' + news[i] + '</li>';
     }
     console.log(lis);
     var ul = document.querySelector('ul');
     ul.innerHTML = lis;
    }
</script>
</body>
</html>

后端代码如下:

router.js('/getNews',function (req,res) {
    var news = [
        "第11日前瞻:中国冲击4金 博尔特再战200米羽球",
        "正直播柴飚/洪炜出战 男双力争会师决赛",
        "女排将死磕巴西!郎平安排男陪练模仿对方核心",
        "没有中国选手和巨星的110米栏 我们还看吗?",
        "中英上演奥运金牌大战",
        "博彩赔率挺中国夺回第二纽约时报:中国因对手服禁药而丢失的奖牌最多",
        "最“出柜”奥运?同性之爱闪耀里约",
        "下跪拜谢与洪荒之力一样 都是真情流露"
    ];
    var data = []
    for(var i=0; i<3 ;i++){
        var index = parseInt(Math.random() * news.length);
        data.push(news[index]);
        news.splice(index,1);
    }
    res.header('Access-Control-Allow-Origin', '*');
    res.send(JSON.stringify(data));
});

3.降域
原理是将主域名相同,子域名不同的域名都降低为主域名。这个时候域名相同,浏览器的同源策略就不会生效了。适用于在同一个页面中不同子域名之间的相互访问。
a.html代码:

<!doctype html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport"
          content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>Document</title>
    <style>
        .ct{
            width: 910px;
            margin: auto;
        }
        .main{
            float: left;
            width: 450px;
            height: 300px;
            border: 1px solid #ccc;
        }
        .main input{
            margin: 20px;
            width: 200px;
        }
        .iframe{
            float: right;
        }
        iframe{
            width: 450px;
            height: 300px;
            border: 1px dashed #ccc;
        }
    </style>
</head>
<body>
<div class="ct">
    <h1>使用降域实现跨域</h1>
    <div class="main">
        <input type="text" placeholder="http://a.jinlong.com:8080/a.html">
    </div>

    <iframe src="http://localhost:8080/b.html" frameborder="0" ></iframe>

</div>

<script>
    document.querySelector('.main input').addEventListener('input',function () {
        window.frames[0].document.querySelector('input').value = this.value;
    });
    document.domain = 'jinlong.com';
</script>
</body>
</html>

b.html代码:

<!doctype html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport"
          content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>Document</title>
    <style>
        html,body{
            margin: 0;
        }
        input{
            margin: 20px;
            width: 200px;
        }
    </style>

</head>
<body>
<input type="text"  placeholder="http://b.jinlong.com:8080/b.html">

<script>
    document.querySelector('input').addEventListener('input',function () {
        window.parent.document.querySelector('.main input').value = this.value;
    });
    document.domain = 'jinlong.com';
</script>

</body>
</html>

4.PostMessage
postMessage比较适用于跨源通信,它的实现条件只需要协议、端口号、主机相同即可,对url无要求。所以在同一页面中,不同的url窗口,可以相互传递和接收数据。
a.html代码:

<!doctype html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport"
          content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>Document</title>
    <style>
        .ct{
            width: 910px;
            margin: auto;
        }
        .main{
            float: left;
            width: 450px;
            height: 300px;
            border: 1px solid #ccc;
        }
        .main input{
            margin: 20px;
            width: 200px;
        }
        .iframe{
            float: right;
        }
        iframe{
            width: 450px;
            height: 300px;
            border: 1px dashed #ccc;
        }
    </style>
</head>
<body>
<div class="ct">
    <h1>使用降域实现跨域</h1>
    <div class="main">
        <input type="text" placeholder="http://a.jinlong.com:8080/a.html">
    </div>

    <iframe src="http://localhost:8080/b.html" frameborder="0" ></iframe>

</div>

<script>
    document.querySelector('.main input').addEventListener('input',function () {
        window.frames[0].postMessage(this.value,'*');
    });
    window.addEventListener('message',function (e) {
        document.querySelector('.main input').value = e.data;
    })
</script>
</body>
</html>

b.html的代码:

<!doctype html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport"
          content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>Document</title>
    <style>
        html,body{
            margin: 0;
        }
        input{
            margin: 20px;
            width: 200px;
        }
    </style>

</head>
<body>
<input type="text"  placeholder="http://b.jinlong.com:8080/b.html">

<script>
    document.querySelector('input').addEventListener('input',function () {
        window.parent.postMessage(this.value,'*');
    });
    window.addEventListener('message',function (e) {
        document.querySelector('input').value = e.data;
    })
</script>

</body>
</html>
上一篇 下一篇

猜你喜欢

热点阅读