说一说javascript跨域和jsonp

同源策略

在浏览器的安全策略中“同源策略”非常如雷贯耳,说的是协议、域名、端口相同则视为同源,域名也可换成IP地址,不同源的页面脚本不能获取对方的数据。

要是想使用XMLHttpRequest或者常规的AJAX请求获取另一个站点的数据,浏览器会告诉你“XXXX is not allowed by Access-Control-Allow-Orign”.

因为同源策略的存在,防止了跨域访问的安全问题,但同时也损失了方便获取资源的便利。

跨域的src属性

世事又非绝对,浏览器还是允许几个元素跨域访问外部资源的,如:<script>,<img>,<iframe>,也就是说,在html元素中拥有src属性的元素是可以跨域访问资源。

通过src属性,img可以引用其它站点或图床的图片,大大降低本站的图片持久。

通过src属性,script可以引用CDN的JS文件,加速了浏览器的脚本文件的下载,跨域的数据获取更加高效和方便。

通过src属性,iframe可以嵌入其它站点的页面,可以让页面的框架和可变内容分离,内容引用较为灵活,方便引用其它站点,虽然现在越来越不建议使用它。

正因为跨域访问的存在,web世界才能更加的精彩。

而JSONP正式利用了script标签的跨域能力。

JSONP

全名JSON with padding

就是通过约定,访问跨域服务器上数据的方法。

这种约定其实就是一个函数定义,并且具备数据参数的定义,由跨域服务器的脚本或动态生成的脚本调用并且传递数据参数。

该函数称之为“跨域回调函数”。

1.一个简单的跨域脚本调用

本地服务器上的一个脚本,直接引用了跨域服务器上的脚本文件。

<html>
<head>
    <title>test</title>
</head>
<body>
    <script src="http://localhost:3001/javascripts/jsonpsrc1.js"></script>
</body>
</html>

跨域服务器上的脚本jsonpsrc1.js

alert(‘hi 我们不是一个域的哦‘);

运行后,浏览器访问本地服务器上的页面,会弹出alert

2.跨域传递数据

在本地服务器的脚本中,约定一个函数,名为jsonpCallback跨域回调函数交由跨域服务器上的脚本调用。

注意,jsonpCallback跨域回调函数的定义必须是在引用跨域服务器上的脚本之前。

<html>
<head>
    <title>test</title>
</head>
<body>
<script>
    function jsonpCallback(data){
        alert(JSON.stringify(data,null,2));
    }
</script>
<script src="http://localhost:3001/javascripts/jsonpsrc2.js"></script>
</body>
</html>

跨域服务器上的脚本jsonpsrc2.js,按照约定的跨域回调函数名调用,并传递一个数据对象。

jsonpCallback({
    name:‘白色的海‘,
    age:90
});

运行后,浏览器访问本地服务器上的页面,会弹出alert,显示跨域服务器传递过来的数据。

3.动态跨域回调函数

通过上面的2种方法已经基本实现了JSONP的使用,但是还存在一个问题,就是必须要提前约定这个跨域回调函数的名字 。

每次都要按照跨域服务器上的的回调函数名进行定义,极为不便。

那么两个跨域的脚本引用能否不绑定同一个函数名称呢。

可以在本地服务器脚本中任意定义跨域回调函数的名称,将该函数名用过参数请求给跨域服务器,在跨域服务器后台代码上动态拼接生成回调函数的调用字符串并响应给请求方。

在这里是用node.js+express的运行环境,并且以一个简单的查询航班信息的跨域请求进行简单的实现。

在本地服务器页面脚本中定义了showFlightInfo跨域回调函数,用于显示跨域服务器返回的航班信息。

在其后的script标签中对跨域服务器进行了一个JS的请求,并将航班信息和跨域回调函数名带过去。

<html>
<head>
    <title>test</title>
    <link rel="stylesheet" href="/stylesheets/style.css">
</head>
<body>
    <h1>航班信息</h1>
    <h1>某航</h1>
    <script>
        function showFlightInfo(data){
            var flightNoEle = document.createElement(‘h4‘);
            flightNoEle.innerHTML=data.flightNo;

            var fromEle = document.createElement(‘h4‘);
            fromEle.innerHTML=data.from;

            var toEle = document.createElement(‘h4‘);
            toEle.innerHTML=data.to;

            document.body.appendChild(flightNoEle);
            document.body.appendChild(fromEle);
            document.body.appendChild(toEle);
        }
    </script>
    <script src="http://localhost:3001/info/flight?flightNo=MU531&callbackFunc=showFlightInfo"></script>
</body>
</html>

跨域服务器的后台代码中响应JS请求,接收航班号,生成航班信息,连同跨域回调函数名称拼接成JS字符串后响应请求方。

router.get(‘/info/flight‘, function (req, res) {
    //生成航班信息,这里直接拿请求过来的航班号写一个航班信息对象
    var data = {
        flightNo: req.query.flightNo,
        from: ‘北京‘,
        to: ‘上海‘};

    //获取请求过来的跨域回调函数名称
    var callbackFunc = req.query.callbackFunc;

    //拼成一个JS字符串
    var s = callbackFunc + ‘(‘ + JSON.stringify(data, null, 2) + ‘);‘;
    console.log(s);

    //通过设置http的header,告知请求方响应的内容是JS脚本
    res.setHeader(‘content-type‘, ‘text/javascirpt;charset=utf-8‘);
    //res.setHeader(‘content-language‘, ‘zh-CN‘)
    res.write(s);
    res.end();

});

运行后,浏览器访问本地服务器上的页面,会显示跨域服务器上传递过来的航班信息。

为了看得清,将跨域服务器收到的请求和响应的内容输出到控制台,方便查看。

第一行是收到的请求,后面的是对请求的响应,可以看到跨域服务端返回给请求方一个JS的脚本,其中的回调函数名称是请求方定制的。

如此完成JSONP的基本实现。

不只是JSONP

通过跨域请求获取数据本质上是利用了<script>标签的src属性,通过浏览器将跨域的脚本拉过来并同时执行。

那么既然是拉过来执行,那和本页面中的其它脚本的执行没多大区分,那就当然也可以通过生成html元素和css的方式改变页面的显示内容,以实现更丰富的功能,比如广告。

基于上面的航班信息,我要在其中显示一块广告内容,内容来自于跨域服务器。

通过<script>标签引用跨域服务器上对应的广告资源地址

<html>
<head>
    <title>test</title>
    <link rel="stylesheet" href="/stylesheets/style.css">
</head>
    <body>
        <h1>航班信息</h1>
        <script src="http://localhost:3001/ad"></script>
        <h1>某航</h1>
        <script>
            function showFlightInfo(data){
                var flightNoEle = document.createElement(‘h4‘);
                flightNoEle.innerHTML=data.flightNo;

                var fromEle = document.createElement(‘h4‘);
                fromEle.innerHTML=data.from;

                var toEle = document.createElement(‘h4‘);
                toEle.innerHTML=data.to;

                document.body.appendChild(flightNoEle);
                document.body.appendChild(fromEle);
                document.body.appendChild(toEle);
            }
        </script>
        <script src="http://localhost:3001/info/flight?flightNo=MU531&amp;callbackFunc=showFlightInfo"></script>
    </body>
</html>

跨域服务器的后台代码中响应JS请求,生成一串使用document.write向页面中生成html元素的字符串。

router.get(‘/ad‘, function (req, res) {
    //拼接一JS字符串,完成向html页面中输出html标记
    var s = ‘document.write(\‘<div style="background-color:red;width:10rem;height:10rem">我是旅行社广告</div>\‘);‘;
    console.log(s);

    res.setHeader(‘content-type‘, ‘text/javascirpt;charset=utf-8‘);
    res.write(s);
    res.end();
});

运行后,浏览器访问本地服务器上的页面,会在航班信息下方显示红色广告位,而这个广告位完全是跨域服务器生成的并且包括样式。

在此基础上,可以做出很多效果,包括将页面某一部分的生成交给专门的或者对应的业务服务器上完成。

时间: 2024-12-04 16:52:47

说一说javascript跨域和jsonp的相关文章

jQuery(三) javascript跨域问题(JSONP解决)

加油~ --WH 一.什么是javascript跨域问题? 域:服务器域名,唯一标识(协议,域名,端口)必须保证一致,说明域相同 跨域:在一个服务器上,去访问另一个服务器上,并且得到另一个服务器返回回来的值,这就是javascript跨域,其实简单点,之前我们做的ajax,都是在同域中访问,现在只是访问的服务器变成了另外的,不是同一台了.仅此而已.但是这样一变,之前的代码就不能用了. 二.解决javascript跨域问题 解决该问题,有很多种方式,我百度了一下,好像这就属于前端的范畴了,所以我决

javascript 跨域问题 jsonp

转载:http://www.cnblogs.com/choon/p/5393682.html demo 用动态创建<script></script>节点的方式实现了跨域HTTP请求,给<script>标签的src属性中的URL添加一个参数来指定回调函数的名称 服务端: 1 2 3 4 5 6 7 8 9 10 11 public void ProcessRequest(HttpContext context) {     context.Response.Content

关于javascript跨域及JSONP的原理与应用

同源策略,它是由Netscape提出的一个著名的安全策略,现在所有的可支持javascript的浏览器都会使用这个策略. 为什么需要同源策略,这里举个例子: 假设现在没有同源策略,会发生什么事情呢?大家知道,JavaScript可以做很多东西,比如:读取/修改网页中某个值.恩,你现在打开了浏览器,在一 个tab窗口中打开了银行网站,在另外一个tab窗口中打开了一个恶意网站,而那个恶意网站挂了一个的专门修改银行信息的JavaScript,当你访问 这个恶意网站并且执行它JavaScript时,你的

利用javascript跨域访问cookie之广告推广

在上一篇<说一说javascript跨域和jsonp>中,利用JSONP进行了跨域的数据访问,利用JS本身的跨域能力在远端生成HTML结构的方式完成了一个小广告. 在实际应用中, 跨域使用天气预报组件可以使用上面的方式实现,另外一种常用的就是显示某些电商的广告,此广告中会滚动您访问过的产品或者关联想推介给你的产品. 比如在某个A网页中显示了两种广告: 某东的广告,里面显示的东西,都是访问过滴,并且加推了相关的东西 某宝的广告,基本一样呈现方式. 当访问某东某宝的商品时,会把信息放到cookie

Ajax 跨域请求 jsonp获取json数据

遇到Ajax的跨域请求出问题 找了中解决办法如下: 参考内容:http://justcoding.iteye.com/blog/1366102 由于受到浏览器的限制,该方法不允许跨域通信.如果尝试从不同的域请求数据,会出现安全错误.如果能控制数 据驻留的远程服务器并且每个请求都前往同一域,就可以避免这些安全错误.但是,如果仅停留在自己的服务器上,Web 应用程序还有什么用处呢?如果需要从多个第三方服务器收集数据时,又该怎么办? 理解同源策略 同源策略阻止从一个域上加载的脚本获取或操作另一个域上的

AJAX的跨域与JSONP

AJAX的跨域与JSONP 什么是AJAX的跨域请求 出于安全的考虑,如果你要从www.a.com通过Ajax来请求另外一个网站www.b.com的内容,浏览器是不允许你这样做的(不理解这里的安全是指什么?想想如果没有这个限制的话,黑客可以做些什么).那什么样的情况下算是跨域?域名不同那当然算是跨域了,例如a.com向b.com发送请求,这当然就是跨域了,不允许的.不过子域名不同(例如sub.a.com向www.a.com发送请求)甚至是同域名不同端口(例如a.com:80向a.com:8080

JavaScript跨域

JavaScript跨域 js跨域是指通过js在不同的域之间进行数据传输或通信,比如用ajax向一个不同的域请求数据,或者通过js获取页面中不同域的框架中(iframe)的数据.只要协议.域名.端口有任何一个不同,都被当作是不同的域. 同源策略阻止从一个域上加载的脚本获取或操作另一个域上的文档属性.也就是说,受到请求的 URL 的域必须与当前 Web 页面的域相同.这意味着浏览器隔离来自不同源的内容,以防止它们之间的操作. URL 说明 是否允许通信 http://www.a.com/a.jsh

javascript 跨域解决方案

1 用jsonp $.getJSON(" http://www.jinanwuliangye.com/document!searchJSONResult.action?name1="+value1+"&jsoncallback=?", function(json){ if(json.属性名==值){ // 执行代码 } }); $.getJSON(" http://www.tongxinglong.com/document!searchJSONRe

JavaScript 跨域漫游

前言: 最近在公司做了几个项目都涉及到了iframe,也就是在这些iframe多次嵌套的项目中,我发现之前对iframe的认识还是比较不足的,所以就静下心来,好好整理总结了iframe的相关知识:<Iframe 功能详解>. 在做公司项目的过程中,让我纠结之一的就是iframe的跨域问题,在网上查到的资料,基本都是给个思路,加个DEMO,也没有完整的解决方案.所以这里我结合公司的项目实际需求,从新整理了一下javaScript跨域的相关方法. PS:请转载的童鞋,请注明出处 ... 目录: 一