Jsonp 關(guān)鍵字詳解及json和jsonp的區(qū)別,ajax和jsonp的區(qū)別
前言
第一次聽(tīng)說(shuō)jsonp,其實(shí)早在2年之前。當(dāng)時(shí)在做一個(gè)活動(dòng)頁(yè)面的抽獎(jiǎng)模塊,要從服務(wù)端get一個(gè)概率,當(dāng)時(shí)什么都不懂,同事說(shuō)用ajax,我就用ajax,同事說(shuō)dataType改成jsonp,我就改成jsonp。于是乎活動(dòng)頁(yè)面做完了,以后也沒(méi)有碰到過(guò)jsonp,在這期間我一直以為jsonp跟ajax息息相關(guān),是xhr的一種特殊的跨域形式...直到一個(gè)月前的一次面試,問(wèn)到j(luò)sonp我被虐成狗,才決定看下jsonp,好吧,原來(lái)jsonp也不是很難。
為什么要用jsonp?
相信大家對(duì)跨域一定不陌生,對(duì)同源策略也同樣熟悉。什么,你沒(méi)聽(tīng)過(guò)?沒(méi)關(guān)系,既然是深入淺出,那就從頭說(shuō)起。
假如我寫(xiě)了個(gè)index頁(yè)面,頁(yè)面里有個(gè)請(qǐng)求,請(qǐng)求的是一個(gè)json數(shù)據(jù)(不知道json數(shù)據(jù)的猛戳JSON簡(jiǎn)介以及用法匯總),簡(jiǎn)單思考寫(xiě)下如下代碼:
<script src='http://libs.baidu.com/jquery/2.0.0/jquery.min.js'></script>
<script type="text/javascript">
$.ajax({
url: 'http://localhost/a.json',
dataType: "json",
success: function (data) {
console.log(data);
}
})
</script>
{
"name": "hanzichi",
"age": 10
}
樓主把兩個(gè)文件都放在wamp下的www文件夾下,ajax請(qǐng)求沒(méi)有跨域,完美得到結(jié)果:

但是如果我的json文件和index文件不在一個(gè)域下,即跨域(不懂跨域的可參考JavaScript 的同源策略)了呢?
試著在wamp下新開(kāi)個(gè)apache端口(不知道怎么開(kāi)的可參考WampServer下使用多端口訪問(wèn)),將json文件放到該服務(wù)端口的文件夾下(樓主設(shè)置的端口號(hào)為8080,默認(rèn)的是80端口),試著發(fā)送請(qǐng)求:
<script src='http://libs.baidu.com/jquery/2.0.0/jquery.min.js'></script>
<script type="text/javascript">
$.ajax({
url: 'http://localhost:8080/a.json',
dataType: "json",
success: function (data) {
console.log(data);
}
})
</script>

很顯然,提示跨域了!怎么搞?這時(shí)jsonp就要出馬了!
神奇的script標(biāo)簽
與jsonp息息相關(guān)的是script標(biāo)簽,而xhr或者說(shuō)傳統(tǒng)意義上的ajax與之沒(méi)有半毛錢(qián)關(guān)系!
接著看上面的index.html代碼,我們看到頁(yè)面引用了百度cdn的jquery路徑,對(duì)于這樣的方式我們似乎已經(jīng)習(xí)以為常,但是仔細(xì)一想,script標(biāo)簽可是完完全全的跨域的啊...沒(méi)錯(cuò),jsonp的實(shí)現(xiàn)核心就是利用script標(biāo)簽的跨域能力!于是我們靈機(jī)一動(dòng),似乎可以這么搞,動(dòng)態(tài)生成一個(gè)script標(biāo)簽,把json的url賦值給script的src屬性,然后再把這個(gè)script標(biāo)簽插入dom里...
<body>
<script src='http://libs.baidu.com/jquery/2.0.0/jquery.min.js'></script>
<script type="text/javascript">
var s = document.createElement('script');
s.src = 'http://localhost:8080/a.json';
document.body.appendChild(s);
</script>
</body>
我們創(chuàng)建了一個(gè)script標(biāo)簽,而標(biāo)簽內(nèi)包裹的內(nèi)容正是需要的json數(shù)據(jù),但是報(bào)錯(cuò)如下:

原因是因?yàn)閖son數(shù)據(jù)并不是合法的js語(yǔ)句,把上面的json數(shù)據(jù)放在一個(gè)回調(diào)函數(shù)中是最簡(jiǎn)單的方法:
<body>
<script src='http://libs.baidu.com/jquery/2.0.0/jquery.min.js'></script>
<script type="text/javascript">
function jsonpcallback(json) {
console.log(json);
}
var s = document.createElement('script');
s.src = 'http://localhost:8080/a.json';
document.body.appendChild(s);
</script>
</body>
jsonpcallback({
"name": "hanzichi",
"age": 10
});

當(dāng)然,這時(shí)的a.json文件并不一定要這樣命名,改成a.js也不會(huì)有一點(diǎn)問(wèn)題。
而如果是與服務(wù)端交互也是一樣的道理,比如和php:
<body>
<script src='http://libs.baidu.com/jquery/2.0.0/jquery.min.js'></script>
<script type="text/javascript">
function jsonpcallback(json) {
console.log(json);
}
var s = document.createElement('script');
s.src="http://localhost:8080/test.php?callback=jsonpcallback";
document.body.appendChild(s);
</script>
</body>
<?php
$jsondata = '{
"name": "hanzichi",
"age": 10
}';
echo $_GET['callback'].'('.$jsondata.')';
?>
需要注意的是,jsonp提供的url(即動(dòng)態(tài)生成的script標(biāo)簽的src),無(wú)論看上去是什么形式,最終生成返回的都是一段js代碼。
JQuery對(duì)jsonp的封裝
為了便于開(kāi)發(fā),jq對(duì)jsonp也進(jìn)行了封裝,封裝在了ajax方法中。
<script src='http://libs.baidu.com/jquery/2.0.0/jquery.min.js'></script>
<script type="text/javascript">
$.ajax({
url: 'http://localhost:8080/a.json',
dataType: 'jsonp',
jsonpCallback: 'CallBack',
success: function (data) {
console.log(data);
}
});
</script>
CallBack({
"name": "hanzichi",
"age": 10
});
以上代碼是針對(duì)請(qǐng)求文件中寫(xiě)死了callback函數(shù)名的情況。因?yàn)檎?qǐng)求的是json文件,json不是服務(wù)器端的動(dòng)態(tài)語(yǔ)言不能進(jìn)行解析,如果是php或者其他的服務(wù)器端語(yǔ)言,則不用寫(xiě)死函數(shù)名,比如下面這樣:
<script src='http://libs.baidu.com/jquery/2.0.0/jquery.min.js'></script>
<script type="text/javascript">
$.ajax({
url: 'http://localhost:8080/test.php',
dataType: 'jsonp',
success: function (data) {
console.log(data);
}
});
</script>
<?php
$jsondata = '{
"name": "hanzichi",
"age": 10
}';
echo $_GET['callback'].'('.$jsondata.')';
?>
當(dāng)然類(lèi)似的封裝好的方法還有幾種:
// 1
$.getJSON("http://localhost:8080/test.php?callback=?", function(data) {
console.log(data);
});
// 2
$.get('http://localhost:8080/test.php', function(data) {
console.log(data);
}, 'jsonp');
需要注意的是getJSON方法的請(qǐng)求地址url需要帶上callback=?,因?yàn)閖q對(duì)該方法進(jìn)行封裝的時(shí)候并沒(méi)有默認(rèn)回調(diào)函數(shù)變量名為callback,于是php中$_GET['callback']就找不到變量值了。
而一般的jq方法url 中不用指定 callback 參數(shù)。對(duì)于 jQuery 中的 jsonp 來(lái)說(shuō),callback 參數(shù)是自動(dòng)添加的。默認(rèn)情況下,jQuery 生成的 jsonp 請(qǐng)求中 callback 參數(shù)是形如 callback=jQuery200023559735575690866_1434954892929 這種根據(jù)看似隨機(jī)的名字,對(duì)應(yīng)的就是 success 那個(gè)處理函數(shù),所以一般不用特意處理。二如果要寫(xiě)死callback名的話,可以參照上文。

總結(jié)
由于同源策略的限制,XmlHttpRequest只允許請(qǐng)求當(dāng)前源(域名、協(xié)議、端口)的資源,為了實(shí)現(xiàn)跨域請(qǐng)求,可以通過(guò)script標(biāo)簽實(shí)現(xiàn)跨域請(qǐng)求,然后在服務(wù)端輸出JSON數(shù)據(jù)并執(zhí)行回調(diào)函數(shù),從而解決了跨域的數(shù)據(jù)請(qǐng)求,這就是jsonp的核心。
jsonp原理:
1.首先在客戶(hù)端注冊(cè)一個(gè)callback, 然后把callback的名字傳給服務(wù)器。
2.服務(wù)器先生成 json 數(shù)據(jù)。 然后以 javascript 語(yǔ)法的方式,生成一個(gè)function , function 名字就是傳遞上來(lái)的參數(shù) jsonp. 最后將 json 數(shù)據(jù)直接以入?yún)⒌姆绞剑胖玫?function 中,這樣就生成了一段 js 語(yǔ)法的文檔,返回給客戶(hù)端。
3.客戶(hù)端瀏覽器,解析script標(biāo)簽,并執(zhí)行返回的 javascript 文檔,此時(shí)數(shù)據(jù)作為參數(shù),傳入到了客戶(hù)端預(yù)先定義好的 callback 函數(shù)里.(動(dòng)態(tài)執(zhí)行回調(diào)函數(shù))
json和jsonp的區(qū)別,ajax和jsonp的區(qū)別
json和jsonp雖然只有一個(gè)字母的區(qū)別,但是它們之間扯不上關(guān)系。
json是一種輕量級(jí)的數(shù)據(jù)交換格式。
jsonp是一種跨域數(shù)據(jù)交互協(xié)議。
json的優(yōu)點(diǎn):(1)基于純文本傳遞極其簡(jiǎn)單,(2)輕量級(jí)數(shù)據(jù)格式適合互聯(lián)網(wǎng)傳遞,(3)容易編寫(xiě)和解析。
ajax和jsonp的區(qū)別:
相同點(diǎn):都是請(qǐng)求一個(gè)url
不同點(diǎn):ajax的核心是通過(guò)xmlHttpRequest獲取內(nèi)容
jsonp的核心則是動(dòng)態(tài)添加<script>標(biāo)簽來(lái)調(diào)用服務(wù)器 提供的js腳本。
相關(guān)文章
JS圖片懶加載的優(yōu)點(diǎn)及實(shí)現(xiàn)原理
這篇文章主要介紹了JS圖片懶加載的優(yōu)點(diǎn)及實(shí)現(xiàn)原理,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-01-01
基于canvas實(shí)現(xiàn)的鐘擺效果完整實(shí)例
這篇文章主要介紹了基于canvas實(shí)現(xiàn)的鐘擺效果,以完整實(shí)例形式分析了JavaScript結(jié)合html5的canvas技術(shù)實(shí)現(xiàn)鐘擺動(dòng)態(tài)旋轉(zhuǎn)效果的方法,需要的朋友可以參考下2016-01-01
微信小程序如何實(shí)現(xiàn)精確的日期時(shí)間選擇器
這篇文章主要介紹了微信小程序如何實(shí)現(xiàn)精確的日期時(shí)間選擇器,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-01-01
一文總結(jié)JavaScript中常見(jiàn)的設(shè)計(jì)模式
在程序設(shè)計(jì)中有很多實(shí)用的設(shè)計(jì)模式,而其中大部分語(yǔ)言的實(shí)現(xiàn)都是基于“類(lèi)”。在程序設(shè)計(jì)中有很多實(shí)用的設(shè)計(jì)模式,而其中大部分語(yǔ)言的實(shí)現(xiàn)都是基于“類(lèi)”。,本文將總結(jié)了JavaScript中常見(jiàn)的十五種設(shè)計(jì)模式,感興趣的朋友可以參考下2023-05-05
js操作CheckBoxList實(shí)現(xiàn)全選/反選(在客服端完成)
對(duì)于CheckBoxList控件來(lái)說(shuō),一方面要實(shí)現(xiàn)大量數(shù)據(jù)在服務(wù)器端的綁定工作,另一方面往往要求實(shí)現(xiàn)全選、反選等功能,接下來(lái)將介紹js操作CheckBoxList實(shí)現(xiàn)全選/反選,感興趣的朋友可以了解下,或許對(duì)你有所幫助2013-02-02
JavaScript函數(shù)參數(shù)使用帶參數(shù)名的方式賦值傳入的方法
這篇文章主要介紹了JavaScript函數(shù)參數(shù)使用帶參數(shù)名的方式賦值傳入的方法,實(shí)例分析了javascript函數(shù)傳遞參數(shù)的使用技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-03-03

