摘要:運行一下頁面,成功彈出提示窗口,的執(zhí)行全過程順利完成到這里為止的話,相信你已經(jīng)能夠理解的客戶端實現(xiàn)原理了吧剩下的就是如何把代碼封裝一下,以便于與用戶界面交互,從而實現(xiàn)多次和重復調用。
先說說JSONP是怎么產(chǎn)生的:
其實網(wǎng)上關于JSONP的講解有很多,但卻千篇一律,而且云里霧里,對于很多剛接觸的人來講理解起來有些困難,小可不才,試著用自己的方式來闡釋一下這個問題,看看是否有幫助。
1、一個眾所周知的問題,Ajax直接請求普通文件存在跨域無權限訪問的問題,甭管你是靜態(tài)頁面、動態(tài)網(wǎng)頁、web服務、WCF,只要是跨域請求,一律不準;
2、不過我們又發(fā)現(xiàn),Web頁面上調用js文件時則不受是否跨域的影響(不僅如此,我們還發(fā)現(xiàn)凡是擁有"src"這個屬性的標簽都擁有跨域的能力,比如、、);
3、于是可以判斷,當前階段如果想通過純web端(ActiveX控件、服務端代理、屬于未來的HTML5之Websocket等方式不算)跨域訪問數(shù)據(jù)就只有一種可能,那就是在遠程服務器上設法把數(shù)據(jù)裝進js格式的文件里,供客戶端調用和進一步處理;
4、恰巧我們已經(jīng)知道有一種叫做JSON的純字符數(shù)據(jù)格式可以簡潔的描述復雜數(shù)據(jù),更妙的是JSON還被js原生支持,所以在客戶端幾乎可以隨心所欲的處理這種格式的數(shù)據(jù);
5、這樣子解決方案就呼之欲出了,web客戶端通過與調用腳本一模一樣的方式,來調用跨域服務器上動態(tài)生成的js格式文件(一般以JSON為后綴),顯而易見,服務器之所以要動態(tài)生成JSON文件,目的就在于把客戶端需要的數(shù)據(jù)裝入進去。
6、客戶端在對JSON文件調用成功之后,也就獲得了自己所需的數(shù)據(jù),剩下的就是按照自己需求進行處理和展現(xiàn)了,這種獲取遠程數(shù)據(jù)的方式看起來非常像AJAX,但其實并不一樣。
7、為了便于客戶端使用數(shù)據(jù),逐漸形成了一種非正式傳輸協(xié)議,人們把它稱作JSONP,該協(xié)議的一個要點就是允許用戶傳遞一個callback參數(shù)給服務端,然后服務端返回數(shù)據(jù)時會將這個callback參數(shù)作為函數(shù)名來包裹住JSON數(shù)據(jù),這樣客戶端就可以隨意定制自己的函數(shù)來自動處理返回數(shù)據(jù)了。
如果對于callback參數(shù)如何使用還有些模糊的話,我們后面會有具體的實例來講解。
JSONP的客戶端具體實現(xiàn):不管jQuery也好,extjs也罷,又或者是其他支持jsonp的框架,他們幕后所做的工作都是一樣的,下面我來循序漸進的說明一下jsonp在客戶端的實現(xiàn):
1、我們知道,哪怕跨域js文件中的代碼(當然指符合web腳本安全策略的),web頁面也是可以無條件執(zhí)行的。
遠程服務器remoteserver.com根目錄下有個remote.js文件代碼如下:
alert("我是遠程文件");
本地服務器localserver.com下有個jsonp.html頁面代碼如下:
毫無疑問,頁面將會彈出一個提示窗體,顯示跨域調用成功。
2、現(xiàn)在我們在jsonp.html頁面定義一個函數(shù),然后在遠程remote.js中傳入數(shù)據(jù)進行調用。
jsonp.html頁面代碼如下:
remote.js文件代碼如下:
localHandler({"result":"我是遠程js帶來的數(shù)據(jù)"});
運行之后查看結果,頁面成功彈出提示窗口,顯示本地函數(shù)被跨域的遠程js調用成功,并且還接收到了遠程js帶來的數(shù)據(jù)。很欣喜,跨域遠程獲取數(shù)據(jù)的目的基本實現(xiàn)了,但是又一個問題出現(xiàn)了,我怎么讓遠程js知道它應該調用的本地函數(shù)叫什么名字呢?畢竟是jsonp的服務者都要面對很多服務對象,而這些服務對象各自的本地函數(shù)都不相同???我們接著往下看。
3、聰明的開發(fā)者很容易想到,只要服務端提供的js腳本是動態(tài)生成的就行了唄,這樣調用者可以傳一個參數(shù)過去告訴服務端“我想要一段調用XXX函數(shù)的js代碼,請你返回給我”,于是服務器就可以按照客戶端的需求來生成js腳本并響應了。
看jsonp.html頁面的代碼:
這次的代碼變化比較大,不再直接把遠程js文件寫死,而是編碼實現(xiàn)動態(tài)查詢,而這也正是jsonp客戶端實現(xiàn)的核心部分,本例中的重點也就在于如何完成jsonp調用的全過程。
我們看到調用的url中傳遞了一個code參數(shù),告訴服務器我要查的是CA1998次航班的信息,而callback參數(shù)則告訴服務器,我的本地回調函數(shù)叫做flightHandler,所以請把查詢結果傳入這個函數(shù)中進行調用。
OK,服務器很聰明,這個叫做flightResult.aspx的頁面生成了一段這樣的代碼提供給jsonp.html(服務端的實現(xiàn)這里就不演示了,與你選用的語言無關,說到底就是拼接字符串):
flightHandler({ "code": "CA1998", "price": 1780, "tickets": 5 });
我們看到,傳遞給flightHandler函數(shù)的是一個json,它描述了航班的基本信息。運行一下頁面,成功彈出提示窗口,jsonp的執(zhí)行全過程順利完成!
4、到這里為止的話,相信你已經(jīng)能夠理解jsonp的客戶端實現(xiàn)原理了吧?剩下的就是如何把代碼封裝一下,以便于與用戶界面交互,從而實現(xiàn)多次和重復調用。
什么?你用的是jQuery,想知道jQuery如何實現(xiàn)jsonp調用?好吧,那我就好人做到底,再給你一段jQuery使用jsonp的代碼(我們依然沿用上面那個航班信息查詢的例子,假定返回jsonp結果不變):
Untitled Page
是不是有點奇怪?為什么我這次沒有寫flightHandler這個函數(shù)呢?而且竟然也運行成功了!哈哈,這就是jQuery的功勞了,jquery在處理jsonp類型的ajax時(還是忍不住吐槽,雖然jquery也把jsonp歸入了ajax,但其實它們真的不是一回事兒),自動幫你生成回調函數(shù)并把數(shù)據(jù)取出來供success屬性方法來調用,是不是很爽呀?
文章版權歸作者所有,未經(jīng)允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉載請注明本文地址:http://systransis.cn/yun/89301.html
摘要:同源策略限制了我們無法通過原生的對象獲取到數(shù)據(jù)。的原理其實不復雜瀏覽器的同源策略把跨域請求都禁止了的標簽是例外,可以突破同源策略從其他來源獲取數(shù)據(jù)由上可得,我們可以通過標簽引入文件,然后通過一系列操作獲取數(shù)據(jù)。上面三點便是實現(xiàn)跨域的原理。 今天做頁面時,后臺給了個接口:https://a.a.com/a/a.json,我頁面的上線地址是:http://b.b.com。顯而易見,因為瀏覽...
摘要:同源策略在中有一個很重要的安全性限制,被稱為同源策略。然而,當進行一些比較深入的前端編程的時候,不可避免地需要進行跨域操作,這時候同源策略就顯得過于苛刻。 JSONP原理 JSON和JSONP JSON(JavaScript Object Notation)是一種輕量級的數(shù)據(jù)交換格式。對于JSON大家應該是很了解了吧,不是很清楚的朋友可以去json.org上了解下,簡單易懂。 ...
摘要:概述是一種跨域通信的手段,它的原理其實很簡單首先是利用標簽的屬性來實現(xiàn)跨域??煽康膶崿F(xiàn)添加回調函數(shù)拼接傳遞的是一個匿名的回調函數(shù),要執(zhí)行的話,暴露為一個全局方法出錯處理使用示例來源個人博客 1. 概述 jsonp是一種跨域通信的手段,它的原理其實很簡單: 首先是利用script標簽的src屬性來實現(xiàn)跨域。 通過將前端方法作為參數(shù)傳遞到服務器端,然后由服務器端注入?yún)?shù)之后再返回,實現(xiàn)服...
摘要:什么叫是填充式或參數(shù)式的簡寫,是通過請求跨域接口,獲取數(shù)據(jù)的新實現(xiàn)方式的實現(xiàn)原理動態(tài)創(chuàng)建標簽,因為標簽是沒有同源策略限制,可以跨域的。具體看接下來的實現(xiàn)這個是庫的具體實現(xiàn),建議下載來研究一下,最好自己動手寫一遍。 什么叫jsonp? jsonp是json with padding(填充式json或參數(shù)式json)的簡寫,是通過ajax請求跨域接口,獲取數(shù)據(jù)的新實現(xiàn)方式 jsonp的實現(xiàn)...
閱讀 1042·2021-09-30 09:58
閱讀 2878·2021-09-09 11:55
閱讀 2035·2021-09-01 11:41
閱讀 1021·2019-08-30 15:55
閱讀 3383·2019-08-30 12:50
閱讀 3528·2019-08-29 18:37
閱讀 3327·2019-08-29 16:37
閱讀 2042·2019-08-29 13:00