Jsonp 關(guān)鍵字詳解及json和jsonp的區(qū)別,ajax和jsonp的區(qū)別
前言
第一次聽說jsonp,其實早在2年之前。當(dāng)時在做一個活動頁面的抽獎模塊,要從服務(wù)端get一個概率,當(dāng)時什么都不懂,同事說用ajax,我就用ajax,同事說dataType改成jsonp,我就改成jsonp。于是乎活動頁面做完了,以后也沒有碰到過jsonp,在這期間我一直以為jsonp跟ajax息息相關(guān),是xhr的一種特殊的跨域形式...直到一個月前的一次面試,問到j(luò)sonp我被虐成狗,才決定看下jsonp,好吧,原來jsonp也不是很難。
為什么要用jsonp?
相信大家對跨域一定不陌生,對同源策略也同樣熟悉。什么,你沒聽過?沒關(guān)系,既然是深入淺出,那就從頭說起。
假如我寫了個index頁面,頁面里有個請求,請求的是一個json數(shù)據(jù)(不知道json數(shù)據(jù)的猛戳JSON簡介以及用法匯總),簡單思考寫下如下代碼:
<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
}
樓主把兩個文件都放在wamp下的www文件夾下,ajax請求沒有跨域,完美得到結(jié)果:

但是如果我的json文件和index文件不在一個域下,即跨域(不懂跨域的可參考JavaScript 的同源策略)了呢?
試著在wamp下新開個apache端口(不知道怎么開的可參考WampServer下使用多端口訪問),將json文件放到該服務(wù)端口的文件夾下(樓主設(shè)置的端口號為8080,默認(rèn)的是80端口),試著發(fā)送請求:
<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>

很顯然,提示跨域了!怎么搞?這時jsonp就要出馬了!
神奇的script標(biāo)簽
與jsonp息息相關(guān)的是script標(biāo)簽,而xhr或者說傳統(tǒng)意義上的ajax與之沒有半毛錢關(guān)系!
接著看上面的index.html代碼,我們看到頁面引用了百度cdn的jquery路徑,對于這樣的方式我們似乎已經(jīng)習(xí)以為常,但是仔細(xì)一想,script標(biāo)簽可是完完全全的跨域的啊...沒錯,jsonp的實現(xiàn)核心就是利用script標(biāo)簽的跨域能力!于是我們靈機一動,似乎可以這么搞,動態(tài)生成一個script標(biāo)簽,把json的url賦值給script的src屬性,然后再把這個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)建了一個script標(biāo)簽,而標(biāo)簽內(nèi)包裹的內(nèi)容正是需要的json數(shù)據(jù),但是報錯如下:

原因是因為json數(shù)據(jù)并不是合法的js語句,把上面的json數(shù)據(jù)放在一個回調(diào)函數(shù)中是最簡單的方法:
<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)然,這時的a.json文件并不一定要這樣命名,改成a.js也不會有一點問題。
而如果是與服務(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(即動態(tài)生成的script標(biāo)簽的src),無論看上去是什么形式,最終生成返回的都是一段js代碼。
JQuery對jsonp的封裝
為了便于開發(fā),jq對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
});
以上代碼是針對請求文件中寫死了callback函數(shù)名的情況。因為請求的是json文件,json不是服務(wù)器端的動態(tài)語言不能進(jìn)行解析,如果是php或者其他的服務(wù)器端語言,則不用寫死函數(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)然類似的封裝好的方法還有幾種:
// 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方法的請求地址url需要帶上callback=?,因為jq對該方法進(jìn)行封裝的時候并沒有默認(rèn)回調(diào)函數(shù)變量名為callback,于是php中$_GET['callback']就找不到變量值了。
而一般的jq方法url 中不用指定 callback 參數(shù)。對于 jQuery 中的 jsonp 來說,callback 參數(shù)是自動添加的。默認(rèn)情況下,jQuery 生成的 jsonp 請求中 callback 參數(shù)是形如 callback=jQuery200023559735575690866_1434954892929 這種根據(jù)看似隨機的名字,對應(yīng)的就是 success 那個處理函數(shù),所以一般不用特意處理。二如果要寫死callback名的話,可以參照上文。

總結(jié)
由于同源策略的限制,XmlHttpRequest只允許請求當(dāng)前源(域名、協(xié)議、端口)的資源,為了實現(xiàn)跨域請求,可以通過script標(biāo)簽實現(xiàn)跨域請求,然后在服務(wù)端輸出JSON數(shù)據(jù)并執(zhí)行回調(diào)函數(shù),從而解決了跨域的數(shù)據(jù)請求,這就是jsonp的核心。
jsonp原理:
1.首先在客戶端注冊一個callback, 然后把callback的名字傳給服務(wù)器。
2.服務(wù)器先生成 json 數(shù)據(jù)。 然后以 javascript 語法的方式,生成一個function , function 名字就是傳遞上來的參數(shù) jsonp. 最后將 json 數(shù)據(jù)直接以入?yún)⒌姆绞?,放置?function 中,這樣就生成了一段 js 語法的文檔,返回給客戶端。
3.客戶端瀏覽器,解析script標(biāo)簽,并執(zhí)行返回的 javascript 文檔,此時數(shù)據(jù)作為參數(shù),傳入到了客戶端預(yù)先定義好的 callback 函數(shù)里.(動態(tài)執(zhí)行回調(diào)函數(shù))
json和jsonp的區(qū)別,ajax和jsonp的區(qū)別
json和jsonp雖然只有一個字母的區(qū)別,但是它們之間扯不上關(guān)系。
json是一種輕量級的數(shù)據(jù)交換格式。
jsonp是一種跨域數(shù)據(jù)交互協(xié)議。
json的優(yōu)點:(1)基于純文本傳遞極其簡單,(2)輕量級數(shù)據(jù)格式適合互聯(lián)網(wǎng)傳遞,(3)容易編寫和解析。
ajax和jsonp的區(qū)別:
相同點:都是請求一個url
不同點:ajax的核心是通過xmlHttpRequest獲取內(nèi)容
jsonp的核心則是動態(tài)添加<script>標(biāo)簽來調(diào)用服務(wù)器 提供的js腳本。
相關(guān)文章
webpack 插件html-webpack-plugin的具體使用
本篇文章主要介紹了webpack 插件html-webpack-plugin的具體使用,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-04-04
javascript 使用sleep函數(shù)的常見方法詳解
這篇文章主要介紹了javascript 使用sleep函數(shù)的常見方法,結(jié)合實例形式分析總結(jié)了javascript sleep函數(shù)的功能、常見使用方法與操作注意事項,需要的朋友可以參考下2020-04-04
JS使用AudioContext實現(xiàn)音頻流實時播放
這篇文章主要為大家詳細(xì)介紹了JavaScript如何使用AudioContext實現(xiàn)音頻流實時播放功能,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-01-01
JavaScript快速排序(quickSort)算法的實現(xiàn)方法總結(jié)
快速排序的思想式 分治法,選一個基準(zhǔn)點,然后根據(jù)大小進(jìn)行分配,分配然完畢之后,對已經(jīng)分配的進(jìn)行遞歸操作,最終形成快速排序,所以遞歸也是快速排序思想的一個重要組成部分,本文主要給大家介紹了JavaScript實現(xiàn)快速排序的寫法,需要的朋友可以參考下2023-11-11

