亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

解決jQuery使用JSONP時產生的錯誤

2019-11-20 11:06:01
字體:
來源:轉載
供稿:網友

什么是域,簡單來說就是協議+域名或地址+端口,3者只要有任何一個不同就表示不在同一個域??缬颍褪窃谝粋€域中訪問另一個域的數據。

如果只是加載另一個域的內容,而不需要訪問其中的數據的話,跨域是很簡單的,比如使用iframe。但如果需要從另一個域加載并使用這些數據的話,就會比較麻煩。為了安全性,瀏覽器對這種情況有著嚴格的限制,需要在客戶端和服務端同時做一些設置才能實現跨域請求。

JSONP簡介
JSONP(JSON with Padding)是一種常用的跨域手段,但只支持JS腳本和JSON格式的數據。顧名思義,JSONP是利用JSON作為墊片,從而實現跨域請求的一種技術手段。其基本原理是利用HTML的<script>標簽天生可以跨域這一特點,用其加載另一個域的JSON數據,加載完成后會自動運行一個回調函數通知調用者。此過程需要另一個域的服務端支持,所以這種方式實現的跨域并不是任意的。

JQuery對JSONP的支持
JQuery的Ajax對象支持JSONP方式的跨域請求,方法是將crossDomain參數指定為true并且將dataType參數指定為jsonp[1],或者使用簡寫形式:getJSON()方法[2]。例如:

// 設置crossDomain和dataType參數以使用JSONP$.ajax({ dataType: "jsonp", url: "http://www.example.com/xxx", crossDomain: true, data: {   }}).done(function() { // 請求完成時的處理函數});// 使用getJSON$.getJSON("http://www.example.com/xxx?jsoncallback=?", { // 參數}, function() { // 請求完成時的處理函數});

使用getJSON時,需要在參數中指定jsoncallback=?,這個就是前面所說的回調函數,JQuery會自動以一個隨機生成的值(回調函數名)來替換該參數中的問號部分,從而形成jsoncallback=jQueryxxxxxxx這種形式的參數,然后和其他參數一起使用GET方式發出請求。

使用第一種方式時,只要將dataType參數的值指定為jsonp,JQuery就會自動在請求地址后面加上jsoncallback參數,因此無需手動添加。

JQuery跨域請求的缺陷:錯誤處理
跨域請求可能會失敗,比如對方服務器的安全設置拒絕接受來自我方的請求(我方不在對方的信任列表中),或者網絡不通,或對方服務器已關閉,或者請求地址或參數不正確導致服務器報錯等等。

在JQuery中,當使用ajax或getJSON發送請求后會返回一個jqXHR對象[3]。該對象實現了Promise協議,所以我們可以使用它的done、fail、always等接口來處理回調。例如我們可以用在它的fail回調中進行請求失敗時的錯誤處理:

var xhr = $.getJSON(...);xhr.fail(function(jqXHR, textStatus, ex) {  alert('request failed, cause: ' + ex.message);});

這種方式能夠處理“正常的錯誤”,例如超時、請求被中止、JSON解析出錯等等。但它對那些“非正常的錯誤”,例如網絡不通、服務器已關閉等情況的支持并不好。

例如當對方服務器無法正常訪問時,在Chrome下你會在控制臺看到一條錯誤信息:

JQuery不會處理該錯誤,而是選擇“靜靜地失敗”:fail回調不會執行,你的代碼也不會得到任何反饋,所以你沒有處理這種錯誤的機會,也無法向用戶報告錯誤。

一個例外是在IE8。在IE8中,當網絡無法訪問時,<script>標簽一樣會返回加載成功的信息,所以JQuery無法根據<script>標簽的狀態來判斷是否已成功加載,但它發現<script>標簽“加載成功”后回調函數卻沒有執行,所以JQuery以此判斷這是一個“解析錯誤”(回調代碼沒有執行,很可能是返回的數據不對導致沒有執行或執行失?。?,因此返回的錯誤信息將是“xxxx was not called”,其中的xxxx為回調函數的名稱。

也就是說,由于IE8(IE7也一樣)的這種奇葩特性,導致在發生網絡不通等“非正常錯誤”時,JQuery反而無法選擇“靜默失敗”策略,于是我們可以由此受益,得到了處理錯誤的機會。例如在這種情況下,上面的例子將會彈出“xxxx was not called”的對話框。

解決方案
當遇到“非正常錯誤”時,除了IE7、8以外,JQuery的JSONP在較新的瀏覽器中全部會“靜默失敗”。但很多時候我們希望能夠捕獲和處理這種錯誤。

實際上在這些瀏覽器中,<script>標簽在遇到這些錯誤時會觸發error事件。例如如果是我們自己來實現JSONP的話可以這樣:

var ele = document.createElement('script');ele.type = "text/javascript";ele.src = '...';ele.onerror = function() {  alert('error');};ele.onload = function() {  alert('load');};document.body.appendChild(ele);

在新瀏覽器中,當發生錯誤時將會觸發error事件,從而執行onerror回調彈出alert對話框:

但是麻煩在于,JQuery不會把這個<script>標簽暴露給我們,所以我們沒有機會為其添加onerror事件處理器。

下面是JQuery實現JSONP的主要代碼:

jQuery.ajaxTransport( "script", function(s) { if ( s.crossDomain ) {  var script,   head = document.head || jQuery("head")[0] || document.documentElement;  return {   send: function( _, callback ) {    script = document.createElement("script");    script.async = true;    ...    script.src = s.url;    script.onload = script.onreadystatechange = ...;    head.insertBefore( script, head.firstChild );   },   abort: function() {    ...   }  }; }});

可以看到script是一個局部變量,從外部無法獲取到。

那有沒有解決辦法呢?當然有:

  • 自己實現JSONP,不使用JQuery提供的
  • 修改JQuery源碼(前提是你不是使用的CDN方式引用的JQuery)
  • 使用本文介紹的技巧

前兩種不說了,如果愿意大可以選擇。下面介紹另一種技巧。

通過以上源碼可以發現,JQuery雖然沒有暴露出script變量,但是它卻“暴露”出了<script>標簽的位置。通過send方法的最后一句:

head.insertBefore( script, head.firstChild );
可以知道這個動態創建的新創建標簽被添加為head的第一個元素。而我們反其道而行之,只要能獲得這個head元素,不就可以獲得這個script了嗎?head是什么呢?繼續看源碼,看head是怎么來的:

head = document.head || jQuery("head")[0] || document.documentElement;
原來如此,我們也用同樣的方法獲取就可以了,所以補全前面的那個例子,如下:

var xhr = $.getJSON(...);// for "normal error" and ie 7, 8xhr.fail(function(jqXHR, textStatus, ex) {  alert('request failed, cause: ' + ex.message);});// for 'abnormal error' in other browsersvar head = document.head || $('head')[0] || document.documentElement; // code from jqueryvar script = $(head).find('script')[0];script.onerror(function(evt) {  alert('error');});

這樣我們就可以在所有瀏覽器(嚴格來說是絕大部分,因為我沒有測試全部瀏覽器)里捕獲到“非正常錯誤”了。

這樣捕獲錯誤還有一個好處:在IE7、8之外的其他瀏覽器中,當發生網絡不通等問題時,JQuery除了會靜默失敗,它還會留下一堆垃圾不去清理,即新創建的<script>標簽和全局回調函數。雖然留在那也沒什么大的危害,但如果能夠順手將其清理掉不是更好嗎?所以我們可以這樣實現onerror:

// handle erroralert('error');// do some clean// delete script nodeif (script.parentNode) {  script.parentNode.removeChild(script);}// delete jsonCallback global functionvar src = script.src || '';var idx = src.indexOf('jsoncallback=');if (idx != -1) {  var idx2 = src.indexOf('&');  if (idx2 == -1) {  idx2 = src.length;  }  var jsonCallback = src.substring(idx + 13, idx2);  delete window[jsonCallback];}

這樣一來就趨于完美了。

完整代碼

function jsonp(url, data, callback) {  var xhr = $.getJSON(url + '?jsoncallback=?', data, callback);  // request failed  xhr.fail(function(jqXHR, textStatus, ex) {    /*     * in ie 8, if service is down (or network occurs an error), the arguments will be:     *      * testStatus: 'parsererror'     * ex.description: 'xxxx was not called' (xxxx is the name of jsoncallback function)     * ex.message: (same as ex.description)     * ex.name: 'Error'     */    alert('failed');  });  // ie 8+, chrome and some other browsers  var head = document.head || $('head')[0] || document.documentElement; // code from jquery  var script = $(head).find('script')[0];  script.onerror = function(evt) {    alert('error');    // do some clean    // delete script node    if (script.parentNode) {      script.parentNode.removeChild(script);    }    // delete jsonCallback global function    var src = script.src || '';    var idx = src.indexOf('jsoncallback=');    if (idx != -1) {      var idx2 = src.indexOf('&');      if (idx2 == -1) {        idx2 = src.length;      }      var jsonCallback = src.substring(idx + 13, idx2);      delete window[jsonCallback];    }  };}

以上代碼在IE8、IE11、Chrome、FireFox、Opera、360下測試通過,其中360是IE內核版本,其他瀏覽器暫時未測。

希望本文對大家學習,幫助大家解決jQuery使用JSONP時產生的錯誤。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
日韩一区二区av| 亚洲精品久久久久中文字幕二区| 久久久久久久亚洲精品| 欧美尤物巨大精品爽| 久久综合亚洲社区| 久久综合亚洲社区| 亚洲精品一区二区在线| 热久久这里只有精品| 亚洲午夜久久久久久久| 国产成人自拍视频在线观看| xxav国产精品美女主播| 亚洲福利视频免费观看| 欧美国产精品va在线观看| 久久久国产一区二区| 91免费综合在线| 成人网在线视频| 精品福利一区二区| 亚洲国产欧美一区二区三区久久| 国产精品99久久久久久人| 91av在线网站| 成人激情春色网| 国产91热爆ts人妖在线| 亚洲性69xxxbbb| 亚洲一区免费网站| 欧美成人精品在线观看| 91av在线免费观看| 亚洲国产私拍精品国模在线观看| 国产精品观看在线亚洲人成网| 国产精品嫩草影院一区二区| 国语自产在线不卡| 26uuu久久噜噜噜噜| 久久视频精品在线| 91精品视频在线免费观看| 国产一区二区成人| 国产精品1234| 国产精品久久二区| 91精品视频在线| 日韩欧美成人网| 国内精品中文字幕| 91精品国产91久久久久福利| 26uuu另类亚洲欧美日本一| 欧美性videos高清精品| 国产精品99久久久久久www| 国产精品丝袜久久久久久不卡| 欧美色图在线视频| 久久99视频免费| 欧美日韩国产91| 亚洲精品国产电影| 国产一区二区三区在线播放免费观看| 国产欧美精品xxxx另类| 亚洲精品自拍偷拍| 亚洲淫片在线视频| 麻豆成人在线看| 动漫精品一区二区| 亚洲欧洲一区二区三区久久| 久久久国产精品亚洲一区| 日韩av在线影视| 久久精品夜夜夜夜夜久久| 亚洲欧美日韩第一区| 亚洲人成在线电影| 高清在线视频日韩欧美| 91系列在线观看| 日韩高清中文字幕| 高清视频欧美一级| 性色av一区二区三区红粉影视| 精品亚洲国产成av人片传媒| 欧美极品少妇xxxxⅹ喷水| 视频在线一区二区| 欧美日韩国产二区| 日韩av网站导航| 欧美第一淫aaasss性| 精品人伦一区二区三区蜜桃网站| 国产欧美日韩高清| 欧美性videos高清精品| 清纯唯美亚洲综合| 69av视频在线播放| 欧美国产精品人人做人人爱| 欧美一区二区三区四区在线| 久久影院资源网| 性欧美办公室18xxxxhd| 欧美激情亚洲国产| 久久亚洲私人国产精品va| 亚洲国产中文字幕在线观看| 成人黄色片网站| 国产91在线播放| 亚洲国产精品久久精品怡红院| 7777精品视频| 在线播放日韩专区| 亚洲第一精品久久忘忧草社区| 一区二区三区黄色| 日韩精品中文字幕在线| 福利视频第一区| 国内精品久久久久影院优| 成人激情视频网| 国产精品稀缺呦系列在线| 91精品国产自产在线观看永久| 日韩美女视频在线观看| 亚洲人成电影网| 欧美国产中文字幕| 国产精品99久久久久久久久久久久| 欧美黄色三级网站| 欧美性xxxx极品hd满灌| 蜜月aⅴ免费一区二区三区| 亚洲欧美日韩爽爽影院| 久久精品电影网| 国产日韩欧美日韩| 久久久999精品视频| 伊人久久精品视频| 国模叶桐国产精品一区| 亚洲亚裔videos黑人hd| 精品久久香蕉国产线看观看亚洲| 国产日韩在线亚洲字幕中文| 国产成人亚洲精品| www.欧美精品| 免费不卡欧美自拍视频| 国产在线拍揄自揄视频不卡99| 国产日韩在线亚洲字幕中文| 亚洲午夜国产成人av电影男同| 国产精品久久久久久av福利软件| 久久99久久99精品中文字幕| 国产精品尤物福利片在线观看| 国产综合在线观看视频| 欧美午夜激情小视频| 日韩中文字幕在线视频播放| 中文在线资源观看视频网站免费不卡| 久久久久久久网站| 欧美亚洲一区在线| 久久久久久久久久久成人| 日本精品一区二区三区在线播放视频| 中文字幕精品www乱入免费视频| 国产精品日韩欧美大师| 91精品国产乱码久久久久久蜜臀| 亚洲国产女人aaa毛片在线| 日韩三级成人av网| 欧美性xxxx极品hd满灌| 国产日韩av在线| 亚洲欧美日本伦理| 日韩精品在线观看一区二区| 欧美成人免费视频| 国产成人精品999| 国产ts人妖一区二区三区| 亚洲欧美制服第一页| 久久久久www| 91免费精品国偷自产在线| 91久久国产婷婷一区二区| 97香蕉超级碰碰久久免费软件| 欧美福利视频在线观看| 欧美日韩久久久久| 尤物精品国产第一福利三区| 欧美色欧美亚洲高清在线视频| 91精品视频在线免费观看| 欧美激情国产高清| 欧美性极品xxxx做受| 亚洲人午夜精品免费| 97色在线视频观看| 亲子乱一区二区三区电影| 国产日本欧美视频| 久久久久久91香蕉国产| 日本免费在线精品| 欧美在线亚洲一区| 国产裸体写真av一区二区| 永久555www成人免费| 国产精品r级在线| 国产精品69精品一区二区三区|