前后端數(shù)據(jù)交互之Ajax原理及使用方法

學習前端到一個階段,必定會碰到一個不可避免的問題,即前端跟后端之間到底是怎么進行數(shù)據(jù)交互的?那么針對這個問題,我們來討論一下以下三種方法:
1、form表單
2、ajax
3、websocket(不討論)

首先,最原始的,通過form表單以post/get方式提交數(shù)據(jù)。

當你點擊submit按鈕時,瀏覽器會默認把你在input里面輸入的數(shù)據(jù),以postget的方式提交到form表單中的action這個地址。相當于你提交表單時,就會向服務器發(fā)送一個請求,然后服務器會接受并處理提交過來的form表單,最后返回一個新的網頁。你可以結合以下代碼來理解這段話。

<form action="/form.html" method="post/get">
    <input type="text" name="username" placeholder="username">
    <input type="password" name="password" placeholder="password">
    <input type="submit">
  </form>

【補充】
1、get提交數(shù)據(jù):請求參數(shù)(一般為input里的值)拼裝成url,相當于向服務器發(fā)url請求。
2、post提交數(shù)據(jù):直接向服務器發(fā)請求,參數(shù)直接發(fā)給后臺。
但是,這種方法會導致幾個問題:
1、在提交時,頁面會發(fā)生跳轉或刷新,導致用戶體驗不好。
2、單項提交,把數(shù)據(jù)提交給后臺,但是不知道后臺會給出怎樣的響應,因為提交后頁面就發(fā)生跳轉了。比如:用戶登錄,那么就不知道到底是注冊成功了還是失敗了。
3、浪費寬帶。因為前后兩個頁面中的大部分HTML代碼往往是相同的。但由于每次應用的交互都需要向服務器發(fā)送請求,應用的響應時間就依賴于服務器的響應時間,這就導致了用戶界面的響應比本地應用慢的多。

為了解決上述問題,2005年出現(xiàn)了Ajax。

一、什么是Ajax
1、Ajax的全稱是Asynchronous JavaScript and XML,即異步JavaScript+XML。
2、它是一種技術方案,但并不是一種新技術。
3、它依賴的是現(xiàn)有的CSS/HTML/Javascript,而其中最核心的依賴是瀏覽器提供的XMLHttpRequest對象。這個對象為向服務器發(fā)送請求和解析服務器響應提供了流暢的接口,使得瀏覽器可以發(fā)出HTTP請求與接收HTTP響應,實現(xiàn)在頁面不刷新的情況下和服務端進行數(shù)據(jù)交互。
【補充】Ajax和XMLHttpRequest 兩者的關系:我們使用XMLHttpRequest對象來發(fā)送一個Ajax請求。
二、怎么實現(xiàn)在頁面不刷新的情況下和服務端進行數(shù)據(jù)交互?
1、XMLHttpRequest對象
2、fecth(不討論)

XMLHttpRequest對象

為了便于我們理解怎么使用XMLHttpRequest對象實現(xiàn)在頁面不刷新的情況下和服務端進行數(shù)據(jù)交互,我們先來看下下面的代碼。

<script>
  var xhr = new XMLHttpRequest()  //創(chuàng)建一個對象
  xhr.open('GET','/helloAjax.json',false)  //設置ajax,.open()方法里面的三個參數(shù)分別是:要發(fā)送的請求類型、請求的url、表示是否異步發(fā)送請求的布爾值
  xhr.send()  //發(fā)出請求
  var data = xhr.responseText  //當請求到來時,讀取請求中的數(shù)據(jù)
  console.log(data)  //輸出
</script>

這樣就是一個使用XMLHttpRequest對象發(fā)送的Ajax請求了,現(xiàn)在我們來分析分析這段代碼。
首先,XMLHttpRequest構造函數(shù)通過new的方式構造一個XHR對象,并將這個對象賦值給xhr(可取任意名字)
然后,調用XMLHttpRequest對象的方法opensend。

XMLHttpRequest對象的兩個重要方法

調用send方法之后請求被發(fā)往服務器,由于這次請求是同步的,JS代碼會在xhr.send()這個步驟暫停掉,一直等到服務器根據(jù)請求生成響應(Response),傳回給XHR對象,再繼續(xù)執(zhí)行。
最后,在收到響應后相應數(shù)據(jù)會填充到XHR對象的屬性。有四個相關屬性會被填充:
1、responseText——從服務器進程作為響應主體被返回的文本。
2、responseXML——從服務器進程返回的DOM兼容的文檔數(shù)據(jù)對象。
3、status——響應的HTTP狀態(tài)。即從服務器返回的數(shù)字代碼,如404(未找到)和200(已就緒)。
4、statusText——HTTP狀態(tài)的說明。伴隨狀態(tài)碼的字符串信息。

但多數(shù)情況下,我們還是要發(fā)送異步請求,才能讓JavaScript繼續(xù)執(zhí)行而不必等待響應。為了更好的理解ajax發(fā)送異步請求,我們來看以下代碼

   <script>
        var xhr = new XMLHttpRequest()
        //請求響應過程的當前活動階段
        xhr.onreadystatechange = function(){
            console.log('readyState:',xhr.readyState)
        }
        xhr.open('GET','hello.json',true)
        xhr.send()
        //監(jiān)聽請求狀態(tài)
        xhr.onload = function(){             //onload相當于readyState=4
            console.log(xhr.status)
            if((xhr.status >= 200 && xhr.status <= 300) || xhr.status === 304){
                console.log(xhr.responseText)
            }else{
                console.log(error)
            }
        }
    </script>

上述代碼中,XHR對象的readyState屬性,表示請求響應過程的當前活動階段。該屬性可取的值如下:

  • 0:未初始化。尚未調用open()方法。
  • 1:啟動。已經調用open()方法,但尚未調用send()方法。
  • 2:發(fā)送。已經調用send()方法,但尚未接收到響應。
  • 3:接收。已經接收到部分響應數(shù)據(jù)。
  • 4:完成。已經接收到全部響應數(shù)據(jù),而且已經可以在客戶端使用了。//onload表示readyState = 4
    【注意】
    1、只要readyState屬性的值由一個值變成另一個值,就會觸發(fā)一次readystatechange事件
    2、必須在調用open()方法之前指定readystatechange事件處理程序才能確??鐬g覽器兼容性。
GET請求/POST請求
  • 與 POST 相比,GET 更簡單也更快,并且在大部分情況下都能用。
  • 然而,在以下情況中,請使用 POST 請求:
    1、無法使用緩存文件(更新服務器上的文件或數(shù)據(jù)庫)
    2、向服務器發(fā)送大量數(shù)據(jù)(POST 沒有數(shù)據(jù)量限制)
    3、發(fā)送包含未知字符的用戶輸入時,POST 比 GET 更穩(wěn)定也更可靠
  • GET請求/POST請求使用方法

一個簡單的 GET 請求

xhr.open('GET','lazyLoad.html',true)
xhr.send()

如果希望通過 GET 方法發(fā)送信息,可以向 URL 末尾添加字符串參數(shù)

xhr.open('GET','/login?username=Iris&password=12345',true)
xhr.send()

一個簡單的 POST 請求

xhr.open('POST','/login',true)
xhr.send('username=Iris&password=12345')

如果向send()里面?zhèn)鬟f一個對象,可以用函數(shù)將該對象拼接成字符串形式

        xhr.open('POST','/login',true)
        xhr.send(makeUrl({             //step3、這個時候send()里面就不用發(fā)字符串了,直接發(fā)傳遞的對象就好了
            username:'Iris',
            address:'ChangSha',
            age:21
        }))

        //step2、用戶傳遞的是一個對象時   (實參是用戶傳遞的這個對象)
        makeUrl({
            username:'Iris',
            address:'ChangSha',
            age:21
        })
        //step1、將拼接的過程寫成一個函數(shù),向函數(shù)makeUrl()里面?zhèn)鬟f一個形參
        function makeUrl(obj){
            var arr = []           //遍歷這個對象
            for(var key in obj){
                arr.push(key + '=' + obj[key])
            }
            return arr.join('&')
        }
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
【社區(qū)內容提示】社區(qū)部分內容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內容

友情鏈接更多精彩內容