一看就懂:jsonp詳解
json相信大家都用的多,jsonp我就一直沒有機(jī)會用到,但也經(jīng)??吹?,只知道是“用來跨域的”,一直不知道具體是個什么東西。今天總算搞明白了。下面一步步來搞清楚jsonp是個什么玩意。
同源策略
首先基于安全的原因,瀏覽器是存在同源策略這個機(jī)制的,同源策略阻止從一個源加載的文檔或腳本獲取或設(shè)置另一個源加載的文檔的屬性??雌饋聿恢朗裁匆馑?,實(shí)踐一下就知道了。
1、隨便建兩個網(wǎng)頁
一個端口是2698,一個2701,按照定義它們是不同源的。

2.用jQuery發(fā)起不同源的請求
在2698端口的網(wǎng)頁上添加一個按鈕,Click事件隨便發(fā)起兩個向端口為2701域的請求。
$("#getOtherDomainThings").click(function () {
$.get("http://localhost:2701/Scripts/jquery-1.4.4.min.js", function (data) {
console.log(data)
})
$.get("http://localhost:2701/home/index", function (data) {
console.log(data)
})
})
根據(jù)同源策略,很明顯會悲劇了。瀏覽器會阻止,根本不會發(fā)起這個請求。(not allowed by Access-Control-Allow-Origin)

OK,原來jsonp是要解決這個問題的。
script標(biāo)簽的跨域能力
不知道大家知不知道CDN這個東西,例如微軟的CDN,使用它,我們的網(wǎng)頁可以不提供jQuery,由微軟的網(wǎng)站幫我們提供:
<script src="http://ajax.aspnetcdn.com/ajax/jquery/jquery-1.8.0.js" type="text/javascript"></script>
回到我們的2698端口的網(wǎng)頁,上面我們在Click事件里有一個對2701端口域的jQuery文件的請求,這次使用script標(biāo)簽來請求。
<script type="text/javascript" src="http://localhost:2701/Scripts/jquery-1.4.4.min.js"></script>
當(dāng)然,200,OK了

同樣是端口2698的網(wǎng)頁發(fā)起對2701域的請求,放在script里設(shè)置scr屬性的OK了,另一個方式就悲劇。利用script的跨域能力,這就是jsonp的基礎(chǔ)。
利用script獲取不同源的json
既然它叫jsonp,很明顯目的還是json,而且是跨域獲取。根據(jù)上面的分析,很容易想到:利用js構(gòu)造一個script標(biāo)簽,把json的url賦給script的scr屬性,把這個script插入到dom里,讓瀏覽器去獲取。實(shí)踐:
function CreateScript(src) {
$("<script><//script>").attr("src", src).appendTo("body")
}
添加一個按鈕事件來測試一下:
$("#getOtherDomainJson").click(function () {
$.get('http://localhost:2701/home/somejson', function (data) {
console.log(data)
})
})

首先,第一個瀏覽器,http://localhost:2701/home/somejson這個Url的確是存在一個json的,而且在 2698網(wǎng)頁上用script標(biāo)簽來請求這個2701這個Url也是200OK的,但是最下面報js語法錯誤了。原來用script標(biāo)簽加載完后,會立即 把響應(yīng)當(dāng)js去執(zhí)行,很明顯{"Email":"zhww@outlook.com","Remark":"我來自遙遠(yuǎn)的東方"}不是合法的js語句。
利用script獲取異域的jsonp
顯然,把上面的json放到一個回調(diào)方法里是最簡單的方法。例如,變成這樣:

如果存在jsonpcallback這個方法,那么jsonpcallback({"Email":"zhww@outlook.com","Remark":"我來自遙遠(yuǎn)的東方"})就是合法的js語句。
由于服務(wù)器不知道客戶端的回調(diào)是什么,不可能hard code成jsonpcallback,所以就帶一個QueryString讓客戶端告訴服務(wù)端,回調(diào)方法是什么,當(dāng)然,QueryString的key要遵從服務(wù)端的約定,上面的是”callback“。
添加回調(diào)函數(shù):
function jsonpcallback(json) {
console.log(json)
}
把前面的方法稍微改改參數(shù):
$("#getJsonpByHand").click(function () {
CreateScript("http://localhost:2701/home/somejsonp?callback=jsonpcallback")
})

200OK,服務(wù)器返回jsonpcallback({"Email":"zhww@outlook.com","Remark":"我來自遙遠(yuǎn)的 東方"}),我們也寫了jsonpcallback方法,當(dāng)然會執(zhí)行。OK順利獲得了json。沒錯,到這里就是jsonp的全部。
利用jQuery獲取jsonp
上面的方式中,又要插入script標(biāo)簽,又要定義一個回調(diào),略顯麻煩,利用jQuery可以直接得到想要的json數(shù)據(jù),同樣是上面的jsonp:
$("#getJsonpByJquery").click(function () {
$.ajax({
url: 'http://localhost:2701/home/somejsonp',
dataType: "jsonp",
jsonp: "callback",
success: function (data) {
console.log(data)
}
})
})
得到的結(jié)果跟上面類似。
總結(jié)
一句話就是利用script標(biāo)簽繞過同源策略,獲得一個類似這樣的數(shù)據(jù),jsonpcallback是頁面存在的回調(diào)方法,參數(shù)就是想得到的json。
jsonpcallback({"Email":"zhww@outlook.com","Remark":"我來自遙遠(yuǎn)的東方"})
ADD 原生js:
<button id="btn">click</button>
<script type="text/javascript">
function $(str){
return document.getElementById(str)
}
function CreateScript(src) {
var Scrip=document.createElement('script');
Scrip.src=src;
document.body.appendChild(Scrip);
}
function jsonpcallback(json) {
console.log(json);//Object { email="中國", email2="中國222"}
}
$('btn').onclick=function(){
CreateScript("http://localhost:51335/somejson?callback=jsonpcallback")
}
</script>
相關(guān)文章
jquery通過select列表選擇框?qū)Ρ砀駭?shù)據(jù)進(jìn)行過濾示例
這篇文章主要介紹了jquery通過select列表選擇框?qū)Ρ砀駭?shù)據(jù)進(jìn)行過濾示例,需要的朋友可以參考下2014-05-05
jquery如何實(shí)現(xiàn)錨點(diǎn)鏈接之間的平滑滾動
實(shí)現(xiàn)錨點(diǎn)鏈接之間的平滑滾動的方法有很多,在接下來的文章中為大家介紹下,jquery是如何實(shí)現(xiàn)的,感興趣的朋友不要錯過2013-12-12
jQuery實(shí)現(xiàn)Meizu魅族官方網(wǎng)站的導(dǎo)航菜單效果
這篇文章主要介紹了jQuery實(shí)現(xiàn)Meizu魅族官方網(wǎng)站的導(dǎo)航菜單效果,可實(shí)現(xiàn)導(dǎo)航菜單項鼠標(biāo)滑過出現(xiàn)下劃線動態(tài)跟隨的功能,通過jQuery插件實(shí)現(xiàn)該功能,需要的朋友可以參考下2015-09-09
jQuery實(shí)現(xiàn)ajax調(diào)用WCF服務(wù)的方法(附帶demo下載)
這篇文章主要介紹了jQuery實(shí)現(xiàn)ajax調(diào)用WCF服務(wù)的方法,以完整實(shí)例形式分析了jQuery的ajax前端調(diào)用及后臺交互調(diào)用WCF服務(wù)的相關(guān)技巧,并附帶完整實(shí)例共讀者下載,需要的朋友可以參考下2015-12-12
jQuery實(shí)用小技巧_輸入框文字獲取和失去焦點(diǎn)的簡單實(shí)例
下面小編就為大家?guī)硪黄猨Query實(shí)用小技巧_輸入框文字獲取和失去焦點(diǎn)的簡單實(shí)例。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-08-08
基于jquery的讓textarea自適應(yīng)高度的插件
jquery extension - auto height text area2010-08-08

