AJAX 跨域请求 - JSONP获取JSON数据

很久没有写随笔了,总是感觉没时间,其实时间就是。。。废话少说,前几天,工作上有一新需求,需要前端web页面异步调用后台的Webservice方法返回信息。实现方法有多种,本例采用jQuery+Ajax,完成后,在本地调试了一切ok,但是部署到服务器上以后就出现问题了,后台服务调用没有响应,怎么回事?代码没怎么改动,唯一修改的地方就是jQuery的ajax方法中的url地址。难道是这里的问题,经过检查和调试,发现原来是同源策略在作怪,我们知道,JavaScript或jQuery是在Web前端开发中经常使用的动态脚本技术。在JavaScript中,有一个很重要的安全性限制,被称为“Same- Origin Policy”(同源策略)。这一策略对于JavaScript代码能够访问的页面内容做了很重要的限制,即JavaScript只能访问与包含它的文档或脚本 在同一域名下的内容。不同域名下的脚本不能互相访问,即便是子域也不行。关于同源策略,读者可百度更详细的解释,这里不再赘述。

 

但是有时候又不可避免地需要进行跨域操作,这时候“同源策略”就是一个限制了,怎么办呢?采用JSONP跨域GET请求是一个常用的解决方案,下面我们来看一下JSONP跨域是如何实现的,并探讨下JSONP跨域的原理。

 

这里提到了JSONP,那有人就问了,它同JSON有什么区别不同和区别呢,接下我们就来看看,百度百科有以下说明:

JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式。它基于JavaScript(Standard ECMA-262 3rd Edition - December 1999)的一个子集。 JSON采用完全独立于语言的文本格式,但是也使用了类似于C语言家族的习惯(包括C, C++, C#, Java, JavaScript, Perl, Python等)。这些特性使JSON成为理想的数据交换语言。易于人阅读和编写,同时也易于机器解析和生成(网络传输速度快)。

JSONP(JSON with Padding)是JSON的 一种“使用模式”,可用于解决主流浏览器的跨域数据访问的问题。由于同源策略,一般来说位于 server1.example.com 的网页无法与不是 server1.example.com的服务器沟通,而 HTML 的  

  • "text/javascript">  
  •     $.getJSON("http://crossdomain.com/services.php?callback=?",  
  •     function(result) {  
  •         for(var i in result) {  
  •             alert(i+":"+result[i]);//循环输出a:1,b:2,etc.  
  •         }  
  •     });  
  •   
  •  

    客户端JS代码在jQuery中的实现方式2:

     

    Js代码   收藏代码
    1. "text/javascript" src="jquery.js">  
    2. "text/javascript">  
    3.     $.ajax({  
    4.         url:"http://crossdomain.com/services.php",  
    5.         dataType:'jsonp',  
    6.         data:'',  
    7.         jsonp:'callback',  
    8.         success:function(result) {  
    9.             for(var i in result) {  
    10.                 alert(i+":"+result[i]);//循环输出a:1,b:2,etc.  
    11.             }  
    12.         },  
    13.         timeout:3000  
    14.     });  
    15.   
     

    客户端JS代码在jQuery中的实现方式3:

     

    Js代码   收藏代码
    1. "text/javascript" src="jquery.js">  
    2. "text/javascript">  
    3.     $.get('http://crossdomain.com/services.php?callback=?', {name: encodeURIComponent('tester')}, function (json) { for(var i in json) alert(i+":"+json[i]); }, 'jsonp');  
    4.   
     

    其中 jsonCallback 是客户端注册的,获取 跨域服务器 上的json数据 后,回调的函数。
    http://crossdomain.com/services.php?callback=jsonpCallback
    这个 url 是跨域服务 器取 json 数据的接口,参数为回调函数的名字,返回的格式为

     

    Js代码   收藏代码
    1. jsonpCallback({msg:'this is json data'})  
     

    Jsonp原理: 
    首先在客户端注册一个callback, 然后把callback的名字传给服务器。

    此时,服务器先生成 json 数据。
    然后以 javascript 语法的方式,生成一个function , function 名字就是传递上来的参数 jsonp.

    最后将 json 数据直接以入参的方式,放置到 function 中,这样就生成了一段 js 语法的文档,返回给客户端。

    客户端浏览器,解析script标签,并执行返回的 javascript 文档,此时数据作为参数,传入到了客户端预先定义好的 callback 函数里.(动态执行回调函数)

     

    使用JSON的优点在于:

    • 比XML轻了很多,没有那么多冗余的东西。
    • JSON也是具有很好的可读性的,但是通常返回的都是压缩过后的。不像XML这样的浏览器可以直接显示,浏览器对于JSON的格式化的显示就需要借助一些插件了。
    • 在JavaScript中处理JSON很简单。
    • 其他语言例如PHP对于JSON的支持也不错。

    JSON也有一些劣势:

    • JSON在服务端语言的支持不像XML那么广泛,不过JSON.org上提供很多语言的库。
    • 如果你使用eval()来解析的话,会容易出现安全问题。

    尽管如此,JSON的优点还是很明显的。他是Ajax数据交互的很理想的数据格式。

     

    主要提示:

    JSONP 是构建 mashup 的强大技术,但不幸的是,它并不是所有跨域通信需求的万灵药。它有一些缺陷,在提交开发资源之前必须认真考虑它们。

     

    第一,也是最重要的一点,没有关于 JSONP 调用的错误处理。如果动态脚本插入有效,就执行调用;如果无效,就静默失败。失败是没有任何提示的。例如,不能从服务器捕捉到 404 错误,也不能取消或重新开始请求。不过,等待一段时间还没有响应的话,就不用理它了。(未来的 jQuery 版本可能有终止 JSONP 请求的特性)。

     

    JSONP 的另一个主要缺陷是被不信任的服务使用时会很危险。因为 JSONP 服务返回打包在函数调用中的 JSON 响应,而函数调用是由浏览器执行的,这使宿主 Web 应用程序更容易受到各类攻击。如果打算使用 JSONP 服务,了解它能造成的威胁非常重要。

    你可能感兴趣的:(javascript)