JS跨域之JSONP方式

2018/4/16 晴天

本人小白一枚,網(wǎng)上關(guān)于jsonp跨域的博文和回答實(shí)在是太多了,看到好多種??jsonp的方式,并沒(méi)有一一去嘗試,可能原理都相同吧,還沒(méi)有做深入了解,只是先記錄一下我所掌握的jsonp解決跨域的方法,希望和我一樣看的頭暈眼花的小伙伴能有所收獲~

1 同源策略

要理解跨域,先要了解“同源策略”,所謂同源是指,域名,協(xié)議,端口相同。所謂“同源策略”,簡(jiǎn)單的說(shuō)就是基于安全考慮,當(dāng)前域不能訪問(wèn)其他域的東西。
一些常見(jiàn)的是否同源示例可參照下表:


image

在同源策略下,在某個(gè)服務(wù)器下的頁(yè)面是無(wú)法獲取到該服務(wù)器以外的數(shù)據(jù)的。例如我們?cè)谧约旱木W(wǎng)站通過(guò)ajax去獲取豆瓣提供的接口數(shù)據(jù)。即使是訪問(wèn)本地非JS文件的文件(比如.json文件).

不過(guò)火狐可以正常讀取到,原因是:雖然允許跨域很不安全,但是如果不跨域的話又帶來(lái)很多不便,所以火狐是允許跨域的。

為什么谷歌獲取不到?

這里要提到一點(diǎn),訪問(wèn)本地計(jì)算機(jī)中的文件,使用的是file協(xié)議,file協(xié)議主要用來(lái)訪問(wèn)本地計(jì)算機(jī)中的文件,就如同在windows資源管理器中打開(kāi)文件一樣,注意它是針對(duì)本地(本機(jī))的,簡(jiǎn)單來(lái)說(shuō),file協(xié)議是訪問(wèn)你本機(jī)的文件資源。
谷歌的報(bào)錯(cuò)可以在控制臺(tái)明顯看到:

Cross origin requests are only supported for protocol schemes:http,data,chrome,chrome-extension,https

實(shí)質(zhì)上就是安全機(jī)制認(rèn)為加載本地其他文件是跨域行為。谷歌瀏覽器會(huì)跨域失敗,是因?yàn)闉g覽器安全機(jī)制不允許。

為什么谷歌不支持跨域?
這是因?yàn)闉g覽器的安全策略,即禁止ajax訪問(wèn)本地的文件,這是不安全也是不允許的,舉例的話,就相當(dāng)于你訪問(wèn)了一個(gè)網(wǎng)站,這個(gè)網(wǎng)站可以讀取到你本地的文件.......

2 跨域問(wèn)題

Chrome真的很?chē)?yán)格,對(duì)于Chrome來(lái)說(shuō),訪問(wèn)本地非js文件或者非同源地址都屬于跨域行為,而火狐可以訪問(wèn)本地文件,但是訪問(wèn)非同源地址時(shí)還是跨域的行為。
這類(lèi)問(wèn)題統(tǒng)稱(chēng)為跨域問(wèn)題。

3 JSON

JSON是一種數(shù)據(jù)存儲(chǔ)格式,不懂得朋友可以自行百度一下~

3 JSONP的原理

JSONP是JSON的一種“使用模式”,可用于解決主流瀏覽器的跨域數(shù)據(jù)訪問(wèn)的問(wèn)題,利用<script>元素的開(kāi)放策略,網(wǎng)頁(yè)可以得到從其他來(lái)源動(dòng)態(tài)產(chǎn)生的JSON資料,而這種使用模式就是所謂的JSONP。用JSONP抓到的資料并不是JSON,而是任意的JAVAScript,用javascript直譯器執(zhí)行而不是用JSON解析器解析。

動(dòng)態(tài)創(chuàng)建<script>標(biāo)簽,設(shè)置其src,回調(diào)函數(shù)也在src中設(shè)置,例如:
<script src="(http://....其他域地址)/(文件地址)?(可以寫(xiě)查詢(xún)內(nèi)容)&(callback=(回調(diào)函數(shù)的名稱(chēng)))"> || <script src="index.json?callback=indexDemo"> 或者 <script src="https://api.douban.com/va/book/search?q=javascript&count=1&callback=handleResponse">

在頁(yè)面中,返回的JSON作為參數(shù)傳入回調(diào)函數(shù)中,我們通過(guò)回調(diào)函數(shù)來(lái)操作數(shù)據(jù)。
這里需要注意的是,JSON文件中的數(shù)據(jù)格式如下:

這里json數(shù)據(jù)前面的名稱(chēng)要和第一條script標(biāo)簽中的callback里的函數(shù)名稱(chēng)一樣 indexDemo({ "hello":"bye" })

用過(guò)該方法的朋友會(huì)發(fā)現(xiàn)JSON文件會(huì)報(bào)錯(cuò),先別著急否定這個(gè)方法,先讀一遍下面這段話~(詭異臉)

根據(jù)上述過(guò)程總結(jié)的話,JSONP的原理如下:

  • 首先在客戶(hù)端注冊(cè)一個(gè)callback,然后把callback的名字傳給服務(wù)器。
  • 此時(shí)服務(wù)器先生成JSON數(shù)據(jù)
  • 然后以JS語(yǔ)法的方式,生成一個(gè)function,function名字就是傳遞上來(lái)的參數(shù)jsonp
  • 最后將json數(shù)據(jù)直接以入?yún)⒌姆绞?,放置到function中,這樣就生成了一段Js語(yǔ)法的文檔,返回給客戶(hù)端。
  • 客戶(hù)端瀏覽器,解析script標(biāo)簽,并執(zhí)行返回的Js文檔,此時(shí)數(shù)據(jù)作為參數(shù),傳入到了客戶(hù)端預(yù)先定義好的callback函數(shù)里。

看到這里,會(huì)不會(huì)有一種煥然大悟的感覺(jué)~原來(lái).json文件中放置的是一個(gè)函數(shù),只有是函數(shù),才可以將json數(shù)據(jù)以入?yún)⒌姆绞?,放置到function中。

4 JSONP的優(yōu)缺點(diǎn)

優(yōu)點(diǎn)

  • 比XML輕了很多,沒(méi)有那么多冗余的東西。
  • JSON具有很好的可讀性,但是通常返回的都是壓縮過(guò)后的,不像XML這樣的瀏覽器可以直接顯示,瀏覽器對(duì)于JSON的格式化顯示就需要借助一些插件了。
  • 在JS中處理JSON很簡(jiǎn)單
  • 其他語(yǔ)言,例如PHP對(duì)于JSON的支持也不錯(cuò)

缺點(diǎn)

  • JSON在服務(wù)端語(yǔ)言的支持不像XML那么廣泛,不過(guò)JSON.org上提供很多語(yǔ)言的庫(kù)(我并不懂這句話,以后再做研究)
  • 如果你使用eval()來(lái)解析的話,會(huì)容易出現(xiàn)安全問(wèn)題(依舊不懂,以后遇到再來(lái)補(bǔ)充~)
5 補(bǔ)充

JSONP是很強(qiáng)大的,但不是所有跨域通信需求的萬(wàn)靈藥,還是建議可以學(xué)會(huì)使用node或其他后端語(yǔ)言來(lái)配合完成跨域的方法~(雖然我也還沒(méi)學(xué)會(huì),學(xué)會(huì)就會(huì)更新在博客里)

第一,也是最重要的一點(diǎn),沒(méi)有關(guān)于JSONP調(diào)用的錯(cuò)誤處理,如果動(dòng)態(tài)腳本插入有效,就執(zhí)行調(diào)用;如果無(wú)效,就靜默失敗。失敗是沒(méi)有任何提示的,服務(wù)器既不會(huì)捕捉到404錯(cuò)誤,也不會(huì)取消或者重新開(kāi)始請(qǐng)求。

第二,被不信任的服務(wù)使用時(shí)會(huì)很危險(xiǎn),因?yàn)镴SONP服務(wù)返回打包在函數(shù)調(diào)用中的JSON相應(yīng),而函數(shù)調(diào)用是由瀏覽器執(zhí)行的,這使宿主web應(yīng)用程序更容易受到各類(lèi)攻擊,如果打算使用JSONP服務(wù),了解它能造成的威脅非常重要。

參考網(wǎng)址
輕松搞定JSONP跨域請(qǐng)求
AJAX跨域請(qǐng)求—JSONP獲取JSON數(shù)據(jù)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容