AJAX javascript的跨域訪問執(zhí)行
更新時間:2008年04月20日 22:40:54 作者:
在網(wǎng)站調(diào)用加載評論等信息的時候遇到了不同域名間javascript不能執(zhí)行的問題,一直都在加載中顯示不出來,而換個網(wǎng)址訪問的話就能正確顯示,一直沒去注意瀏覽器提示的錯誤信息:

突然感覺就是這里的問題,研究一下,搞定后其實覺得挺容易的,只是自己知識還是有些欠缺,解決方法如下:
阻塞的AJAX請求
我們先來證實一下請求的阻塞情況吧。我們使用如下的代碼:
連續(xù)發(fā)起三個請求
復制代碼 代碼如下:
function simpleRequest()
{
var request = new XMLHttpRequest();
request.open(”POST“, “Script.ashx“);
request.send(null);
}
function threeRequests()
{
simpleRequest();
simpleRequest();
simpleRequest();
}
當執(zhí)行threeRequests時就會連續(xù)發(fā)出3個相同域名的請求,還是通過統(tǒng)計圖表來查看阻塞的效果:

最后的請求被前兩個請求阻塞
每個請求需要花費1.5秒的時間。很明顯,第三個請求必須等到第一個請求結(jié)束之后才能執(zhí)行,因此總共需要進行3秒多鐘才能執(zhí)行完畢。我們要改變的就是這個狀況。
傳統(tǒng)的跨域名異步請求解決方案
AJAX安全性的唯一保證,似乎就是對于跨域名(Cross-Domain)AJAX請求 的限制。除非打開本地硬盤的網(wǎng)頁,或者在IE中將跨域名傳輸數(shù)據(jù)的限制打開,否則向其他域名發(fā)出AJAX請求都會被禁止。而且對于跨域名的判斷非常嚴格, 不同的子域名,或者相同域名的不同端口,都會被認作是不同的域名,我們不能向它們的資源發(fā)出AJAX請求。
從表面上看起來似乎沒有辦法打破這個限制,還好我們有個救星,那就是iframe!
iframe雖然不在標準中出現(xiàn),但是由于它實在有用,F(xiàn)ireFox也“不得不”對它進 行了支持(類似的還有innerHTML)。網(wǎng)上已經(jīng)有一些跨域名發(fā)出異步請求的做法,但是它們實在做的不好。它們的簡單工作原理如下:在另一個域名下放 置一個特定的頁面文件作為Proxy,主頁面將異步請求的信息通過Query String傳遞入iframe里的Proxy頁面,Proxy頁面在AJAX請求執(zhí)行完畢后將結(jié)果放在自己location的hash中,而主頁面會對 iframe的src的hash值進行輪詢,一旦發(fā)現(xiàn)它出現(xiàn)了改變,則通過hash值得到需要的信息。
這個方法的實現(xiàn)比較復雜,而且功能有限。在 IE和FireFox中,對于URL的長度大約可以支持2000個左右的字符。對于普通的需求它可能已經(jīng)足夠了,可惜如果真要傳遞大量的數(shù)據(jù),這就遠遠不 夠了。與我們一會兒要提出的解決方案相比,可能它唯一的優(yōu)勢就是能夠跨任意域名進行異步請求,而我們的解決方案只能突破子域名的限制。
那么現(xiàn)在來看看我們的做法!
優(yōu)雅地突破子域名的限制
我們突破子域名限制的關(guān)鍵還是在于iframe。
iframe是的好東西,我們能夠跨過子域名來訪問iframe里的頁面對象,例如 window和DOM結(jié)構(gòu),包括調(diào)用JavaScript(通過window對象)——我們將內(nèi)外頁面的 document.domain設為相同就可以了。然后在不同子域名的頁面發(fā)起不同的請求,把結(jié)果通過JavaScript進行傳遞即可。唯一需要的也僅 僅是一個簡單的靜態(tài)頁面作為Proxy而已。
我們現(xiàn)在就來開始編寫一個原形,雖然簡單,但是可以說明問題。
首先,我們先來編寫一個靜態(tài)頁面,作為放在iframe里的Proxy,如下:
SubDomainProxy.html
復制代碼 代碼如下:
<html xmlns=“http://www.w3.org/1999/xhtml” >
<head>
<title>Untitled Page</title>
<script type=“text/javascript” language=“javascript”>
document.domain = “test.com“; function sendRequest(method, url)
{
var request = new XMLHttpRequest();
request.open(method, url);
request.send(null);
}
</script>
</head>
<body>
</body>
</html>
然后我們再編寫我們的主頁面:
http://www.test.com/Default.html
復制代碼 代碼如下:
<html xmlns=“http://www.w3.org/1999/xhtml” >
<head runat=“server”>
<title>Untitled Page</title>
<script type=“text/javascript” language=“javascript”>
document.domain = “test.com“; function simpleRequest()
{
var request = new XMLHttpRequest();
request.open(”POST“, “Script.ashx“);
request.send(null);
}
function crossSubDomainRequest()
{
var proxy = document.getElementById(”iframeProxy“).contentWindow;
proxy.sendRequest('POST', ‘http://sub0.test.com/Script.ashx‘);
}
function threeRequests()
{
simpleRequest();
simpleRequest();
crossSubDomainRequest();
}
</script>
</head>
<body>
<input type=“button” value=“Request” onclick=“threeRequests()” />
<iframe src=“http://sub0.test.com/SubDomainProxy.html” style=“display:none;”
id=“iframeProxy”></iframe>
</body>
</html>

不同域名的請求不會被阻塞
令人滿意的結(jié)果!
雖說只能突破子域名,但是這已經(jīng)足夠了,不是嗎?我們?yōu)槭裁匆獜娗笕我庥蛎g能夠異步通 訊呢?更何況我們的解決方案是多么的優(yōu)雅!在下一篇文章中,我們將會為ASP.NET AJAX客戶端實現(xiàn)一個完整的CrossSubDomainRequestExecutor,它會自動判斷是否正在發(fā)出跨子域名的請求,并選擇AJAX請 求的方式。這樣,客戶端的異步通訊層就會對開發(fā)人員完全透明。世上還會有比這更令人愉快的事情嗎?:)
注意事項
可能以下幾點值得一提:
我在出現(xiàn)這個想法之后也作了一些嘗試,最后發(fā)現(xiàn)創(chuàng)建XMLHttpRequest對象,調(diào)用open方法和send方法都必須在iframe中的頁面中執(zhí)行才能夠在IE和FireFox中成功發(fā)送AJAX請求。
在上面的例子中,我們向子域名請求的的路徑是http://sub0.test.com/Script.ashx。請注意,完整的子域名不可以省略,否則在FireFox下就會出現(xiàn)權(quán)限不夠的錯誤,在調(diào)用open方法時就會拋出異?!坪鮂ireFox把它當作了父頁面域名的資源了。
因為瀏覽器的安全策略,瀏覽器不允許不同域(比如:phinest.org和lab.phinest.org)、不同協(xié)議(比如: http://phinest.org和https://phinest.org)、不同端口(比如:http: phinest.org和http://phinest.org:8080)下的頁面通過XMLHTTPRequest相互訪問,這個問題同樣影響著不同頁面的Javascript的相互調(diào)用和控制,但是當主域、協(xié)議、端口相同時,通過設置頁面的document.domain主域, Javascript可以在不同的子域名間訪問控制,比如通過設置document.domain='phinest.org',http: //phinest.org和http://lab.phinest.org頁面可互訪,這個特性也提供了此情況下不同子域名下的 XMLHTTPRequest相互訪問的解決方案。
對于主域、協(xié)議、端口相同時的Ajax跨域問題,很早就有設置document.domain來解決的說法,但一直沒有看到具體的成功應用,這次嘗試了一下,其原理就是,利用一個隱藏的iframe引入所跨另一子域的頁面作為代理,通過Javascript來控制iframe引入的另一子域的 XMLHTTPRequest來進行數(shù)據(jù)獲取。對于不同主域/不同協(xié)議/不同端口下的Ajax訪問需要通過后臺的代理來實現(xiàn)。
您可能感興趣的文章:
- 利用nginx解決cookie跨域訪問的方法
- 用jQuery與JSONP輕松解決跨域訪問的問題
- jQuery 跨域訪問問題解決方法
- Ajax實現(xiàn)跨域訪問的三種方法
- Javascript 跨域訪問解決方案
- js iframe跨域訪問(同主域/非同主域)分別深入介紹
- Ajax 設置Access-Control-Allow-Origin實現(xiàn)跨域訪問
- jquery下利用jsonp跨域訪問實現(xiàn)方法
- JQuery Ajax 跨域訪問的解決方案
- 解決AJAX中跨域訪問出現(xiàn)''沒有權(quán)限''的錯誤
- Python的Django應用程序解決AJAX跨域訪問問題的方法
- AJAX的跨域訪問-兩種有效的解決方法介紹
- 深入淺析同源策略和跨域訪問
- 關(guān)于Iframe如何跨域訪問Cookie和Session的解決方法
- js實現(xiàn)跨域訪問的三種方法
- AJax與Jsonp跨域訪問問題小結(jié)
- jquery 跨域訪問問題解決方法(筆記)
- 淺談Ajax跨域Session和跨域訪問
- 解決nginx/apache靜態(tài)資源跨域訪問問題詳解
相關(guān)文章
ajax方式實現(xiàn)注冊功能(提交數(shù)據(jù)到后臺數(shù)據(jù)庫完成交互)
這篇文章主要介紹了ajax方式實現(xiàn)注冊功能,提交數(shù)據(jù)到后臺數(shù)據(jù)庫完成交互,感興趣的小伙伴們可以參考一下2016-08-08ajax實現(xiàn)點擊不同的鏈接讓返回的內(nèi)容顯示在特定div里
過ajax實現(xiàn)在一個web頁面點擊不同的鏈接,然后將返回的結(jié)果顯示在該頁面固定的div里2014-06-06AJAX 請求區(qū)分 $_SERVER[''HTTP_X_REQUESTED_WITH''] 小解
關(guān)于這個內(nèi)容,很多人都有所了解.但從我搜索的內(nèi)容來看,他們只是略微看一下,根本不知道里面到底是什么情況.2011-06-06jquery與php結(jié)合實現(xiàn)AJAX長輪詢(LongPoll)
傳統(tǒng)的AJAX輪詢方式,客服端以用戶定義的時間間隔去服務器上查詢最新的數(shù)據(jù)。種這種拉取數(shù)據(jù)的方式需要很短的時間間隔才能保證數(shù)據(jù)的精確度,但太短的時間間隔客服端會對服務器在短時間內(nèi)發(fā)送出多個請求。2015-10-10ajax JSONP請求處理回調(diào)函數(shù)jsonpCallback區(qū)分大小寫
使用ajax進行 JSONP跨域請求,因為被請求的對方的回調(diào)函數(shù)名稱是無法修改,想到設置AJAX 的JSONP參數(shù)。但是發(fā)現(xiàn)根本不起作用。最后偶然發(fā)現(xiàn) jsonpcallback是區(qū)分大小寫的2013-09-09