jsonp解決跨域問題

個人認(rèn)為, 網(wǎng)上有很多關(guān)于jsonp的講解, 但實在是講得不夠通俗, 所以我想寫這篇文章能夠讓更多人接受這簡單易懂的jsonp.

同源策略

在講jsonp之前, 我們需要先了解什么是同源策略. 同源意思是指協(xié)議, 域名, 端口號相同. 而同源策略就是基于安全考慮, 當(dāng)前源不能訪問不同源的內(nèi)容.

簡單來說, A服務(wù)器想要通過直接請求訪問B服務(wù)器的數(shù)據(jù)的時候, B服務(wù)器的協(xié)議, 域名, 端口必須與A服務(wù)器的相同, 才能正常獲取. 例如. 我直接通過瀏覽器訪問天氣API來獲取深圳的天氣是可以成功獲得json數(shù)據(jù), 而在代碼中使用ajax獲取則因為同源策略獲取失敗.

直接通過瀏覽器輸入網(wǎng)址訪問:

使用瀏覽器直接訪問API

通過代碼去訪問數(shù)據(jù):
代碼如下:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>ajax</title>
</head>
<body>
    <div id="mydiv">
        <button id="btn">點擊發(fā)起ajax請求獲取天氣數(shù)據(jù)</button>
    </div>
</body>
<script type="text/javascript">
    window.onload = function() {

    var oBtn = document.getElementById('btn');

    oBtn.onclick = function() {

        var xhr = new XMLHttpRequest();

        xhr.onreadystatechange = function() {
            if (xhr.readyState == 4 && xhr.status == 200) {
                    alert( xhr.responseText );
            }
        };

        xhr.open('get', 'http://www.weather.com.cn/data/cityinfo/101280601.html', true);
        xhr.send(); 
    };
};
</script>
</html>

運行點擊發(fā)送ajax請求后得到如下錯誤:


運行結(jié)果.png

這就是同源策略產(chǎn)生的問題.

如何解決?

先說一個概念, 訪問非同源的資源被稱為跨域訪問. 而跨域訪問其實有這么幾個特殊情況, 并不是所有的資源都無法跨域訪問, 其實帶src屬性的<script>, <iframe>, <img>標(biāo)簽和帶href的<link>標(biāo)簽都不受跨域限制.

那么知道了這一點, 我們不難想到, 既然js可以跨域引入, 那么我們可以用這個特性來將數(shù)據(jù)攜帶回來. 所以我們可以回歸主題了, jsonp就是利用<script>來實現(xiàn)跨域獲取數(shù)據(jù)的.

以下是一些不影響任何操作的更改, 可以略過, 直接查看jsonp部分

因為jsonp是需要后端配合的, 而且現(xiàn)在豆瓣開發(fā)API已經(jīng)關(guān)閉使用, 我只能自己編寫測試用的服務(wù)器. 和之前一樣的結(jié)果, 直接進行ajax請求數(shù)據(jù)時還是一樣受同源策略限制, 先貼圖展示一下更改.(實際上只是更改了一下訪問的地址, 希望大家不要覺得和之前有任何區(qū)別)

更改的代碼:
將:
xhr.open('get', 'http://www.weather.com.cn/data/cityinfo/101280601.html', true);
改成了:
xhr.open('get', 'http://localhost:8080/weather', true);
效果:

更改服務(wù)器地址后, ajax跨域限制展示

只是更換了訪問的地址, 效果并沒有任何改變. 不會影響接下來的任何操作.

jsonp

jsonp是 json with padding ( 帶填充的json ) 的簡寫. 現(xiàn)在一般前后端的數(shù)據(jù)傳輸格式都是json格式, 原因是可以簡潔描述復(fù)雜數(shù)據(jù), 主要還有一點是被js原生支持.

jsonp實現(xiàn)跨域請求的原理簡單的說, 就是動態(tài)創(chuàng)建<script>標(biāo)簽, 然后利用<script>的src 不受同源策略約束來跨域獲取數(shù)據(jù).

jsonp由兩部分組成: 回調(diào)函數(shù)和數(shù)據(jù). 回調(diào)函數(shù)是當(dāng)響應(yīng)到來時應(yīng)該在頁面中調(diào)用的函數(shù). 回調(diào)函數(shù)的名字一般是在請求中指定的. 而數(shù)據(jù)就是傳入回調(diào)函數(shù)中的json數(shù)據(jù)寄悯。

這些概念是抄來的, 其實看不明白也沒關(guān)系, 個人覺得實例已經(jīng)講得很清楚了, 可以把下面部分看完了再上來結(jié)合文字理解.

現(xiàn)在大家知道了以下兩點:

  • 什么是同源策略限制
  • <script>標(biāo)簽可以無視跨域限制

那么我們可以進行接下來的講解了, 或許大家還不清楚<script>標(biāo)簽可以無視跨域限制是什么意思, 或者是能做到什么, 那么我用代碼來展示給大家看看.
我的服務(wù)器部分代碼內(nèi)容如下:

@RequestMapping(value = "/print")
public String getFunction(){
    return "print()";
}

看不懂沒關(guān)系, 你只要知道訪問http://localhost:8080/print
會得到如下結(jié)果

訪問效果展示

然后我將運行如下代碼:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>從服務(wù)器加載js進行方法的調(diào)用</title>
</head>
<body>
</body>
    <script>
        function print(){
            console.log("我被調(diào)用了啊!!!");
        }
    </script>
    <script src="http://localhost:8080/print"></script>
</html>

運行結(jié)果如下:


從服務(wù)器加載js進行方法的調(diào)用

從如上展示中我們發(fā)現(xiàn), 服務(wù)器中傳過來的字符串如果使用<script>標(biāo)簽進行引入, 會被解析成js代碼. 從而實現(xiàn)在客戶端聲明方法, 通過服務(wù)器傳來的調(diào)用函數(shù)語句來調(diào)用.

大家可能猜到該怎么使用jsonp了.

那么我們開始說通過jsonp進行跨域獲取localhost:8080/weather返回的數(shù)據(jù)的具體實現(xiàn). 先理一下思路.

  1. 我們知道可以通過<script>標(biāo)簽引入js(就算是普通字符串也會被解析成js). 但是我們不可能直接寫個:<script src="http://localhost:8080/weather"></script>
    因為這跟直接在代碼中寫
<script>{"weatherinfo":{"city":"深圳","cityid":"101280601","temp1":"24℃","temp2":"30℃","weather":"陣雨轉(zhuǎn)大雨","img1":"n3.gif","img2":"d9.gif","ptime":"18:00"}}</script>

一樣, 并沒有任何意義. 我們需要的是對可以操作這些數(shù)據(jù), 而不是那它作為js代碼.

  1. 我們知道可以通過<script>標(biāo)簽從服務(wù)器加載js進行方法的調(diào)用, 這個遠程的方法調(diào)用其實方法在客戶端, 服務(wù)器只是傳來了一個調(diào)用的js指令而已.
    這段代碼在客戶端
<script>
    function print(){
        console.log("我被調(diào)用了啊!!!");
    }
</script>

調(diào)用代碼print()是通過<script>從服務(wù)器中加載.

  1. 還知道我們需要的數(shù)據(jù)產(chǎn)自服務(wù)器, 所以我們不妨讓服務(wù)器攜帶著數(shù)據(jù), 在進行方法調(diào)用的時候把數(shù)據(jù)穿給我們. 這樣的話, 服務(wù)器需要知道我們已經(jīng)寫好在客戶端的方法的方法名, 便于調(diào)用.

思路大概是這樣了, 如果看不懂, 是我表述確實不太貼切了, 很抱歉, 但是大家一定還是能學(xué)會, 還是那句話, 看代碼吧, 代碼更容易懂. 所以具體實現(xiàn)如下:
先看服務(wù)端代碼:

@RequestMapping(value = "/weather",method = RequestMethod.GET)
public String getWeatherByJsonp(@RequestParam("callback") String callback){
    return callback + "({\"weatherinfo\":{\"city\":\"深圳\",\"cityid\":\"101280601\",\"temp1\":\"24℃\",\"temp2\":\"30℃\",\"weather\":\"陣雨轉(zhuǎn)大雨\",\"img1\":\"n3.gif\",\"img2\":\"d9.gif\",\"ptime\":\"18:00\"}})";
}

客戶端代碼:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>jsonp的測試樣例</title>
</head>
<body>
</body>
    <script>
        function printData(data){
            var str = "城市:" + data.weatherinfo.city + "\r\n最低溫度:" + data.weatherinfo.temp1 + "\r\n最高溫度:" + data.weatherinfo.temp2;
            console.log(str);
        }
    </script>
    <script src="http://localhost:8080/weather?callback=printData"></script>
</html>

打開客戶端頁面效果展示:


jsonp的測試樣例展示

看效果我們是成功跨域請求到了數(shù)據(jù). 接下來我給大家解釋一下代碼.

  • 客戶端先自定義一個js函數(shù)printData, 這個函數(shù)接受一個參數(shù)data, 這個data就是我們想要通過請求獲取的. 函數(shù)體中對data(也就是我們需要的數(shù)據(jù))進行處理.
  • 但是我們還沒有這個數(shù)據(jù), 需要調(diào)用函數(shù), 并且傳這個數(shù)據(jù)進來, 數(shù)據(jù)沒有, 調(diào)用方法的語句也沒有, 我們知道這兩個都是可以通過<script>標(biāo)簽引入.
  • 所以, 我們在服務(wù)器端, 不僅僅是返回一個數(shù)據(jù)這么簡單, 我們需要把數(shù)據(jù)嵌入到一個調(diào)用函數(shù)的語句里面, 但是由于服務(wù)器端還不知道前端到底寫了什么函數(shù)來處理這些數(shù)據(jù), 也就是前端自定義的函數(shù)名我們還不清楚
  • 所以需要前端通過請求時攜帶參數(shù)callback=printData傳一個函數(shù)名過來告訴服務(wù)器端, 服務(wù)器端就使用這個printData包裹住數(shù)據(jù)返回到客戶端, 從而進行攜帶參數(shù)的調(diào)用了printData(data).

簡而言之就是先假設(shè)有個能拿到數(shù)據(jù)的函數(shù)A, 然后我們通過<script>引入服務(wù)端的調(diào)用函數(shù)A的語句, 這個語句里包含了數(shù)據(jù). 并執(zhí)行了函數(shù)A.

這就是我總結(jié)的jsonp解決跨域問題的過程. 希望能對大家有哪怕一絲幫助.

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市堕义,隨后出現(xiàn)的幾起案子猜旬,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,454評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件洒擦,死亡現(xiàn)場離奇詭異椿争,居然都是意外死亡,警方通過查閱死者的電腦和手機熟嫩,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,553評論 3 385
  • 文/潘曉璐 我一進店門秦踪,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人掸茅,你說我怎么就攤上這事椅邓。” “怎么了昧狮?”我有些...
    開封第一講書人閱讀 157,921評論 0 348
  • 文/不壞的土叔 我叫張陵景馁,是天一觀的道長。 經(jīng)常有香客問我逗鸣,道長裁僧,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,648評論 1 284
  • 正文 為了忘掉前任慕购,我火速辦了婚禮聊疲,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘沪悲。我一直安慰自己获洲,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 65,770評論 6 386
  • 文/花漫 我一把揭開白布殿如。 她就那樣靜靜地躺著贡珊,像睡著了一般。 火紅的嫁衣襯著肌膚如雪涉馁。 梳的紋絲不亂的頭發(fā)上门岔,一...
    開封第一講書人閱讀 49,950評論 1 291
  • 那天,我揣著相機與錄音烤送,去河邊找鬼寒随。 笑死,一個胖子當(dāng)著我的面吹牛帮坚,可吹牛的內(nèi)容都是我干的妻往。 我是一名探鬼主播,決...
    沈念sama閱讀 39,090評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼试和,長吁一口氣:“原來是場噩夢啊……” “哼讯泣!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起阅悍,我...
    開封第一講書人閱讀 37,817評論 0 268
  • 序言:老撾萬榮一對情侶失蹤好渠,失蹤者是張志新(化名)和其女友劉穎昨稼,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體拳锚,經(jīng)...
    沈念sama閱讀 44,275評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡悦昵,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,592評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了晌畅。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片但指。...
    茶點故事閱讀 38,724評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖抗楔,靈堂內(nèi)的尸體忽然破棺而出棋凳,到底是詐尸還是另有隱情,我是刑警寧澤连躏,帶...
    沈念sama閱讀 34,409評論 4 333
  • 正文 年R本政府宣布剩岳,位于F島的核電站,受9級特大地震影響入热,放射性物質(zhì)發(fā)生泄漏拍棕。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 40,052評論 3 316
  • 文/蒙蒙 一勺良、第九天 我趴在偏房一處隱蔽的房頂上張望绰播。 院中可真熱鬧,春花似錦尚困、人聲如沸蠢箩。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,815評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽谬泌。三九已至,卻和暖如春逻谦,著一層夾襖步出監(jiān)牢的瞬間掌实,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,043評論 1 266
  • 我被黑心中介騙來泰國打工邦马, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留贱鼻,地道東北人。 一個月前我還...
    沈念sama閱讀 46,503評論 2 361
  • 正文 我出身青樓勇婴,卻偏偏與公主長得像忱嘹,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子耕渴,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,627評論 2 350

推薦閱讀更多精彩內(nèi)容

  • 首先,需要明確記住的是齿兔,jsonp不是ajax的一種特例橱脸,而是使用動態(tài)script來獲取數(shù)據(jù)的一種方式础米。 原理 由...
    疊搭寶箱閱讀 779評論 0 0
  • JSONP是JSON with Padding的略稱。它是一個 非官方 的協(xié)議添诉,它允許在服務(wù)器端集成Script ...
    李霖弢閱讀 815評論 0 3
  • 為什么會有跨域問題屁桑?因為瀏覽器的同源策略同源策略是什么?協(xié)議栏赴,域名蘑斧,端口都相同如果非同源,會受到什么限制须眷?(1) ...
    07120665a058閱讀 307評論 0 1
  • 1.什么是跨域竖瘾?跨域,指的是瀏覽器不能執(zhí)行其他網(wǎng)站的腳本花颗。它是由瀏覽器的同源策略造成的捕传,是瀏覽器對javascri...
    貝灬小暉閱讀 313評論 0 1
  • http://blog.csdn.net/qq_34482827/article/details/51655914...
    cllian119閱讀 1,032評論 0 0