jsonp解決跨域問題

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

同源策略

在講jsonp之前, 我們需要先了解什么是同源策略. 同源意思是指協(xié)議, 域名, 端口號相同. 而同源策略就是基于安全考慮, 當前源不能訪問不同源的內(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>標簽和帶href的<link>標簽都不受跨域限制.

那么知道了這一點, 我們不難想到, 既然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格式, 原因是可以簡潔描述復雜數(shù)據(jù), 主要還有一點是被js原生支持.

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

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

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

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

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

那么我們可以進行接下來的講解了, 或許大家還不清楚<script>標簽可以無視跨域限制是什么意思, 或者是能做到什么, 那么我用代碼來展示給大家看看.
我的服務(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>標簽進行引入, 會被解析成js代碼. 從而實現(xiàn)在客戶端聲明方法, 通過服務(wù)器傳來的調(diào)用函數(shù)語句來調(diào)用.

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

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

  1. 我們知道可以通過<script>標簽引入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>標簽從服務(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)用.

思路大概是這樣了, 如果看不懂, 是我表述確實不太貼切了, 很抱歉, 但是大家一定還是能學會, 還是那句話, 看代碼吧, 代碼更容易懂. 所以具體實現(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>標簽引入.
  • 所以, 我們在服務(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)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

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

  • 首先,需要明確記住的是,jsonp不是ajax的一種特例,而是使用動態(tài)script來獲取數(shù)據(jù)的一種方式。 原理 由...
    疊搭寶箱閱讀 818評論 0 0
  • 為什么會有跨域問題?因為瀏覽器的同源策略同源策略是什么?協(xié)議,域名,端口都相同如果非同源,會受到什么限制?(1) ...
    07120665a058閱讀 324評論 0 1
  • 1.什么是跨域?跨域,指的是瀏覽器不能執(zhí)行其他網(wǎng)站的腳本。它是由瀏覽器的同源策略造成的,是瀏覽器對javascri...
    貝灬小暉閱讀 323評論 0 1
  • http://blog.csdn.net/qq_34482827/article/details/51655914...
    cllian119閱讀 1,145評論 0 0
  • 大家好,我是一名的學生 簡單的介紹一下自己吧,喜歡干凈的生活,喜歡做精彩的事,驕傲、小帥,性格直接。 雖然是...
    歐痕閱讀 200評論 0 1

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