JSONP跨域模擬百度搜索
一、什么是JSONP
JSONP
是JSON with padding
(填充式JSON或參數(shù)式JSON)的簡(jiǎn)寫(xiě),是應(yīng)用JSON的一種新方法,在后來(lái)的Web服務(wù)中非常流行,JSONP看起來(lái)與JSON差不多,只不過(guò)是被包含在函數(shù)中調(diào)用的JSON,就像下面這樣:
callback({"name": "王歡"});
SONP由兩部分組成:回調(diào)函數(shù)和數(shù)據(jù)?;卣{(diào)函數(shù)是當(dāng)響應(yīng)到來(lái)時(shí)應(yīng)該在頁(yè)面 中調(diào)用的函數(shù)?;卣{(diào)函數(shù)的名字一般是在請(qǐng)求中指定的。而數(shù)據(jù)就是傳入回調(diào)函數(shù)中的JSON數(shù)據(jù)。下面就是一個(gè)典型的JSONP請(qǐng)求。
https://freegeoip.net/json/?callback=handleResponse
這個(gè)URL是在請(qǐng)求一個(gè)JSONP
地理定位服務(wù),通過(guò)查詢(xún)字符串來(lái)指定JSONP服務(wù)的回調(diào)參數(shù)是很常見(jiàn)的,就像上面的URL所示,這里指定的回調(diào)函數(shù)的名字叫做:handleResponse()
JSONP是通過(guò)動(dòng)態(tài)<script>
元素來(lái)使用的,使用時(shí)可以為src屬性指定一個(gè)跨域URL。可以不受限制的從其他域加載資源,因?yàn)镴SONP是有效的JavaScript
代碼,所以在請(qǐng)求完成后,即在JSONP響應(yīng)加載到頁(yè)面中以后,就會(huì)立即執(zhí)行。
二、JSONP跨域請(qǐng)求
我們知道,同源策略是瀏覽器的一種安全機(jī)制,所謂的源是指協(xié)議、域名和端口號(hào),當(dāng)我們的腳本在運(yùn)行時(shí),瀏覽器會(huì)檢測(cè)它所執(zhí)行的腳本和他所取得的的數(shù)據(jù)與我們HTML頁(yè)面是否相同,如果相同,就是同源的,會(huì)進(jìn)行成功的請(qǐng)求,如果他們的源不相同,就是跨域請(qǐng)求。在默認(rèn)情況下,瀏覽器是不支持跨域請(qǐng)求的,那么如果我們想要跨域請(qǐng)求,該如何操作呢?
script
標(biāo)簽是不受同源策略的限制的,即我們?cè)谡?qǐng)求script腳本的時(shí)候,無(wú)論是在HTML所在的服務(wù)器還是其他服務(wù)器,它都可以請(qǐng)求到,所以我們就利用script
標(biāo)簽的這種性質(zhì)來(lái)進(jìn)行數(shù)據(jù)的跨域請(qǐng)求。就來(lái)看看JSONP
是如何進(jìn)行跨域請(qǐng)求的。
首先,我們請(qǐng)求一段script
代碼,這段代碼里如果它能調(diào)用我們所指定的一個(gè)函數(shù),并將數(shù)據(jù)作為實(shí)參傳遞進(jìn)來(lái),那么只要我們定義了這個(gè)函數(shù)并定義了形參,形參就會(huì)接收到他的實(shí)參來(lái)得到數(shù)據(jù)。舉個(gè)例子:
假設(shè)在腳本中定義了一個(gè)getData(data)
,如果現(xiàn)在請(qǐng)求一個(gè)腳本,這個(gè)腳本能夠調(diào)用getData()這個(gè)函數(shù),并將data 作為實(shí)參傳遞進(jìn)來(lái),那形參收到的數(shù)據(jù)就可以進(jìn)行相應(yīng)的處理。
<script> function getData(data){ console.log(data); } var script = document.createElement('script'); script.id = 'jsonp'; script.src = 'jsonp.js'; document.body.appendChild(script); </script>
假設(shè)前端已經(jīng)把函數(shù)名告訴了后端,后端就可以調(diào)用這個(gè)getData()
,并且可以傳遞信息。在jsonp.html
就可以請(qǐng)求到下述jsonp.js
文件。
getData({ name: '小王', age: 20 })
運(yùn)行得到結(jié)果為:
得到了一個(gè)Object
對(duì)象,就是我們所傳遞的數(shù)據(jù)。
那么,我們?nèi)绾胃嬖V服務(wù)器getData()
這個(gè)函數(shù)呢?如果我們每次都固定是getData(),
我們的開(kāi)發(fā)會(huì)很死板,就不能定義其他函數(shù)名字。實(shí)際上,我們可以通過(guò)get請(qǐng)求將我們前端定義的函數(shù)名字通過(guò)參數(shù)告訴后端,后端動(dòng)態(tài)生成這樣的一個(gè)腳本文件并返回給函數(shù)的調(diào)用就可以。
百度就有一個(gè)這樣的接口,我們一起來(lái)看看。
打開(kāi)瀏覽器百度頁(yè)面,打開(kāi)調(diào)試工具,看一下NETwork
標(biāo)簽下面會(huì)監(jiān)聽(tīng)所有關(guān)于瀏覽器向服務(wù)器發(fā)送的http請(qǐng)求并查看數(shù)據(jù)。
在搜索框鍵入“b”,請(qǐng)求如圖:
請(qǐng)求得到的關(guān)鍵字為:
這里的callback
函數(shù)其實(shí)是jquery生成的一個(gè)全局函數(shù)。得到這個(gè)URL
后,我們可以保存其有用的信息,并將回調(diào)函數(shù)換成別的函數(shù):
https://www.baidu.com/sugrec?pre=1&wd=b&req=2&csor=1&cb=getData();
將其輸入到地址欄中進(jìn)行測(cè)試:
可以發(fā)現(xiàn),這個(gè)回調(diào)函數(shù)就變成了我們?cè)O(shè)置的。
三、模擬百度搜索
我們現(xiàn)在就可以通過(guò)這個(gè)接口去發(fā)生JSON來(lái)模擬一下百度搜索頁(yè)面。
我們定義一個(gè)全局變量作為接收數(shù)據(jù)的函數(shù),data就是接收到的數(shù)據(jù),一旦getdata()
函數(shù)被調(diào)用,說(shuō)明我們我們的Jsonp函數(shù)就發(fā)送完成,此時(shí)可以通過(guò)removeChild()
刪除script標(biāo)簽,這樣就可以實(shí)現(xiàn)每當(dāng)我們發(fā)送一次請(qǐng)求,script
標(biāo)簽在接收到數(shù)據(jù)后就會(huì)被刪除,在數(shù)據(jù)處理的時(shí)候,getData()
返回給我們的是一個(gè)對(duì)象,對(duì)象里面有一個(gè)關(guān)鍵字g,里面對(duì)應(yīng)的是一個(gè)數(shù)組,數(shù)組里面是字符串,我們先將這個(gè)數(shù)組遍歷,然后根據(jù)每個(gè)元素生成一個(gè)li加到input下面的ul就可以,先將ul里面的html清空,這樣的話,每次請(qǐng)求的li都是全新的,當(dāng)keyup
出來(lái)的時(shí)候,首先取到當(dāng)前input
的value值并調(diào)用getdata()
函數(shù),將wd傳進(jìn)來(lái),就實(shí)現(xiàn)了jsonp傳遞數(shù)據(jù)的過(guò)程。
代碼如下:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Document</title> <style> div { position: relative; width: 600px; height: 40px; } input { width: 500px; height: 40px; border: 2px solid #4E6EF2; } button{ position: absolute; left: 411px; top: 0; width: 95px; height: 44px; background-color: #4E6EF2; border: none; font-size: 18px; color: white; } ul{ position: relative; left: -40px; top: -10px; width: 411px; height: 400px; } li{ height: 40px; width: 411px; line-height: 40px; font-size: 16px; list-style: none; } </style> </head> <body> <div> <input type="text" value =''> <button>百度一下</button> </div> <ul></ul> <script src="jquery.js"></script> <script> // function getData(data){ var script = document.querySelector('#jsonp'); script.parentNode.removeChild(script); $('ul').html(''); for(var i =0;i<data.g.length;i++){ $('<li>'+data.g[i].q +'</li>').appendTo('ul'); } } //動(dòng)態(tài)生成script腳本 function getList(wd){ var script = document.createElement('script'); script.id = 'jsonp'; script.src = 'https://www.baidu.com/sugrec?pre=1&p=3&ie=utf-8&json=1&prod=pc&from=pc_web&sugsid=26350&req=2&csor=1&cb=getData&wd='+wd; document.body.appendChild(script); } //給 var ipt = document.querySelector('input'); ipt.addEventListener('keyup',function(){ var wd = this.value; getList(wd); console.log(wd); }) </script> </body> </html>
效果為:
四、JSONP缺點(diǎn)
JSONP之所以在開(kāi)發(fā)人員中極為流行,是因?yàn)樗浅:?jiǎn)單易用,不過(guò)他也有兩點(diǎn)不足:
- 首先,
JSONP
是從其他域中加載執(zhí)行代碼。如果其他域不安全,很可能會(huì)在響應(yīng)中夾帶一些惡意代碼,而此時(shí)除了完全放棄JSONP調(diào)用之外,沒(méi)有辦法追究。 其次,要確定JSONP請(qǐng)求是否失敗并不容易。雖然HTML5給
<script>元素新增了一個(gè)onerror事件處理程序,但目前還沒(méi)有得到任何瀏覽器的支持。為此,開(kāi)發(fā)人員不得不使用計(jì)時(shí)器檢測(cè)指定時(shí)間內(nèi)是否接收到了響應(yīng)。但是畢竟不是每個(gè)用戶(hù)的上網(wǎng)速度和帶寬都一樣,所以操作起來(lái)也不盡人意。
到此這篇關(guān)于JSONP跨域模擬百度搜索的文章就介紹到這了,更多相關(guān)JSONP跨域模擬百度搜索內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
JS/jquery實(shí)現(xiàn)一個(gè)網(wǎng)頁(yè)內(nèi)同時(shí)調(diào)用多個(gè)倒計(jì)時(shí)的方法
這篇文章主要介紹了JS/jquery實(shí)現(xiàn)一個(gè)網(wǎng)頁(yè)內(nèi)同時(shí)調(diào)用多個(gè)倒計(jì)時(shí)的方法,涉及js與jQuery基于定時(shí)器的時(shí)間相關(guān)操作技巧,需要的朋友可以參考下2017-04-04Three.JS實(shí)現(xiàn)三維場(chǎng)景
這篇文章主要為大家詳細(xì)介紹了Three.JS實(shí)現(xiàn)三維場(chǎng)景,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-12-12解決微信瀏覽器Javascript無(wú)法使用window.location.reload()刷新頁(yè)面
這篇文章主要介紹在微信瀏覽器Javascript無(wú)法使用window.location.reload()刷新頁(yè)面的解決方法,比較實(shí)用,需要的朋友可以參考下。2016-06-06CocosCreator實(shí)現(xiàn)技能冷卻效果
這篇文章主要介紹了CocosCreator實(shí)現(xiàn)技能冷卻效果,同學(xué)們可以跟著教程,親手試一下,代碼都是可以復(fù)用的2021-04-04js實(shí)現(xiàn)兼容性好的微軟官網(wǎng)導(dǎo)航下拉菜單效果
這篇文章主要介紹了js實(shí)現(xiàn)兼容性好的微軟官網(wǎng)導(dǎo)航下拉菜單效果,涉及JavaScript基于鼠標(biāo)事件實(shí)現(xiàn)頁(yè)面樣式變換的技巧,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2015-09-09layui 彈出層回調(diào)獲取彈出層數(shù)據(jù)的例子
今天小編就為大家分享一篇layui 彈出層回調(diào)獲取彈出層數(shù)據(jù)的例子,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-09-09利用js將ajax獲取到的后臺(tái)數(shù)據(jù)動(dòng)態(tài)加載至網(wǎng)頁(yè)中的方法
今天小編就為大家分享一篇利用js將ajax獲取到的后臺(tái)數(shù)據(jù)動(dòng)態(tài)加載至網(wǎng)頁(yè)中的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-08-08微信小程序scroll-view實(shí)現(xiàn)自定義滾動(dòng)條
這篇文章主要為大家詳細(xì)介紹了微信小程序scroll-view實(shí)現(xiàn)自定義滾動(dòng)條,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-06-06