美文网首页
JSONP_跨域

JSONP_跨域

作者: 山门龙龙 | 来源:发表于2017-07-20 02:30 被阅读0次

1.什么是同源策略

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

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

  • 跨域就是访问非本域的资源
  • 跨域的实现方式:
    1.JSONP
    2.CORS
    3.降域
    4.PostMessage

3.JSONP的原理是什么

  • script标签可以引入其它域的JS,利用这一特性,可以实现跨域访问接口,但前提是后端支持。
  • JSONP的实现过程:
    1.在script标签中定义函数fun
    2.在script标签中的src存放后端接口地址,并在后面添加callback=fun
    3.后端接口经过处理,返回一个fun(data)的JSON字符串。
    4.script接收后端返回的数据,然后fun(data)执行。

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>

相关文章

  • JSONP_跨域

    1、同源策略 同源策略是众多的安全策略之一,是在web层面上的策略。同源策略规定:不同域的客户端脚本在没有明确授权...

  • JSONP_跨域

    题目1: 什么是同源策略 浏览器出于安全方面的考虑,只允许与本域下的接口交互。不同源的客户端脚本在没有明确授权的情...

  • JSONP_跨域

    题目1: 什么是同源策略 浏览器出于安全方面的考虑,只允许与本域下的接口交互。不同源的客户端脚本在没有明确授权的情...

  • JSONP_跨域

    1、什么是同源策略? 同源策略,即Same origin policy,它是所有支持JavaScript的浏览器都...

  • JSONP_跨域

    什么是同源策略 浏览器出于安全方面的考虑,只允许与本域下的接口交互。不同源的客户端脚本在没有明确授权的情况下,不能...

  • JSONP_跨域

    1. 什么是同源策略 浏览器出于安全方面的考虑,只允许与本域下的接口交互。不同源的客户端脚本在没有明确授权的情况下...

  • JSONP_跨域

    1: 什么是同源策略 同源策略(Same origin policy)是一种约定,是浏览器最核心也最基本的安全功能...

  • JSONP_跨域

    1. 什么是同源策略 同源策略(Same origin Policy):浏览器出于安全方面的考虑,只允许与本域下的...

  • JSONP_跨域

    1.什么是同源策略 同源策略(Same origin Policy):浏览器出于安全方面的考虑,只允许与本域下的接...

  • JSONP_跨域

    题目1: 什么是同源策略 浏览器出于安全考虑,只允许与本域下的接口交互。不同源的客户端脚本在没有明确授权的情况下,...

网友评论

      本文标题:JSONP_跨域

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