什麼是 AJAX?
AJAX (Asynchronous JavaScript and XML) 是一種允許網頁在「不重新整理整個頁面」的情況下,透過 JavaScript 與伺服器進行非同步資料交換的技術。由於 JavaScript 是單執行緒運作,若採用同步請求會導致瀏覽器在等待回應時完全卡死;AJAX 解決了這個痛點,讓網頁能只更新局部內容(如更新購物車、載入評論)。雖然名稱包含 XML,但現代開發已全面轉向更輕量的 JSON 格式。從傳統的 XMLHttpRequest (XHR) 到現代內建的 Fetch API,以及強大的第三方庫 Axios,掌握這些 HTTP Client 實作方式是前端工程師處理資料流、優化使用者體驗的核心基本功。
在前面幾篇文章,小編介紹了 什麼是 RESTful API、API 系統設計方法、Web API 實作解析。接下來,這份 AJAX 完全指南 就來介紹怎麼透過 AJAX 這個技術和後端的 Web API 溝通。
以男女交往來說重要的溝通是:
- 丟球: 會不會丟球,不會丟球對方就不知道該怎麼和你進一步互動。
- 接球: 對方丟球該怎麼接到,該怎麼看到球飛過來。
當後端的 Web API 開好後,前端也要能夠知道該怎麼對後端發出請求,也才能夠做到良好的互動。掌握 XMLHttpRequest 或 Fetch API 便是關鍵。
什麼是 AJAX:實現非同步網頁通訊的技術
AJAX(Asynchronous JavaScript And XML),直接用翻譯看就是非同步 JavaScript 與 XML,雖然是寫 XML 但也可以用現今廣泛使用的 JSON 來做資料交換。
因為 JavaScript 單線程一次只能做一件事,在發送請求到伺服器時會需要等待回覆,若這時伺服器一直沒回應,在單線程的情況下其他任務會被卡死,因此也才有了非同步請求。AJAX 可以讓網頁能夠只更新需要的部分,而無須重新載入整個頁面。
XMLHttpRequest:傳統的 AJAX 實作方式
W3C 標準的非同步請求是利用 XMLHttpRequest 寫法:
- 優點: 簡單直觀所有相關操作都從這個物件去取得
- 缺點: 結構會不漂亮,所以後來 jQuery AJAX 以原生的標準為基礎,封裝後讓寫法更好
XMLHttpRequest 是一個能夠取得伺服器端的資料的物件,取得資料的方式可以為非同步 (asynchronously) 或同步 (synchronously)。
發出 request,需叫用 HTTP request 類別的 open() 及 send() 兩個方法
- send(): 發出 request,若為 POST 可以帶入資料
- open(): request 相關設定,有三個參數
- 第一個參數為 HTTP request 的方法 (全大寫)
- GET: 請求資源
- POST: 新增資源
- PUT: 取代資源
- PATCH: 更新資源
- DELETE: 刪除資源
- 第二個參數為 URL: 預設會擋 CORS
- 第三個參數為是否非同步,預設是非同步
- 第一個參數為 HTTP request 的方法 (全大寫)
發出 request 後,要檢查 request 目前的 readyState。
如果狀態值為 4 代表伺服器已經傳回所有資訊了,readyState 所有可能的值如下:
- 0: 還沒開始
- 1: 讀取中
- 2: 已讀取
- 3: 資訊交換中
- 4: 完成
1 | var data = null; |
jQuery AJAX:簡化原生通訊的熱門函式庫
jQuery 的 Ajax 則以原生的標準為基礎,封裝後讓寫法更好,大幅簡化程式且提供多種方法。
- $.get(url,data,callback): get 傳送 data 到 url,並透過 callback 取得執行指令與 url 結果
- $.post(url,data,callback) 同上,但以 post 傳送 data 到 url
- $.ajax(setting) 完整的 Ajax 控制語法,setting 為多資料之物件結構陣列
1 | var settings = { |
Promise:處理非同步操作的現代標準
Promise 是承諾的意思,可以想成不管是否完成都會告知結果。這是在 JavaScript 中處理非同步程式碼的一種新風格樣式。
代表瀏覽器說著:「我承諾我會盡快給予你一個答覆」
Promise 是在 JavaScript 中非同步程式碼的一種新風格樣式,目前已經實作在各大主流瀏覽器中,定義可以看這個網頁。
Promise 建立之後,要不成功要不失敗,一個好例子是 fetch API,它基本上就是新一代的 XMLHttpRequest,另外 Async/Await 是 Promise 下一代的解決方案,把 then 取代成 Async/Await 後,就能按照同步的語法撰寫。
1 | function asyncFunction(value) { |
主要使用 then 及丟進去的 callback 去取值出來,這裡的 callback 就都是非同步執行的喔!!!
1 | asyncFunction("test").then(function (datums) { |
promise.all,適合用在多支 API 要一起執行,並確保全部完成後才進行其他工作時。
1 | Promise.all([promise(1), promise(2), promise(3, 5000)]).then((res) => { |
promise.race,適合用在站點不穩定,同時發送多支同行為 API 確保可行性使用。
1 | Promise.race([promise(1), promise(2), promise(3, 5000)]).then((res) => { |
promise.allSettled 適合有多個不相互依賴的異步任務才能成功完成 (想知道每個承諾的結果),在所有都已完成或拒絕後完成的 Promise。
1 | const promise1 = Promise.resolve(3); |
Async/Await:讓非同步程式碼更簡潔的語法糖
Promise 在設置上可能會有些複雜並難以理解,因此現代瀏覽器實作出 Async 以及 Await。
- async 能夠讓函式執行非同步行為
- await 可以被用在 async 函式內部,讓程式碼繼續執行前去等待一個 Promise 完成
這能讓程式碼在一連串的 Promise 的情況時更加簡潔易懂,只要加上 await 就可以直接叫用一個返回 Promise 的函數,程式碼就會直接在那裏等待,直到 Promise 被完成。
1 | async function fetchProducts() { |
從 Promise 改成 async/await 優化了什麼
1 | function getData() { |
Fetch API:現代瀏覽器內建的新一代請求介面
在這裡我們看到 fetch() 帶一個 URL 參數並回傳一個 Promise,可以看成是 AJAX 的新版本,詳情可看MDN 的簡介。
看寫法的話就知道是已經基於 promise 來設計的,其中要注意的是預設 fetch 不會收或送 cookie 必須設定 credentials 這個參數才可以。
1 | fetch("http://localhost:8080/api/v1/book/1", { method: "get" }) |
Axios:功能強大且易用的 HTTP Client 函式庫
Axios 除了整合上面的功能外,提供了簡化的寫法,是目前前端開發中最受歡迎的 HTTP Client 之一。
1 | axios.get("http://localhost:8080/api/v1/book/1").then((res) => { |
配置基本設定把一些常用的設定或共用一些需要的防呆實作
1 | // 基本設定 |
Vue.js 的文件也有介紹使用這套函式庫,透過配置將 client 和vue 的 instance 綁在一起,可以直接透過this.$http來叫用。
AJAX 打 API 避免快取的實用技巧
將時間或亂數以適當的 ? 或 & 加在 URL 後方就可以做到避免快取且跨瀏覽器相容。
例如:
1 | // 不快取網頁 |
FAQ:AJAX 常見問題
Q1:我該選原生 Fetch 還是 Axios?
A:如果您的專案極小且不希望增加 Bundle Size,Fetch 已經足夠。但對於中大型專案,強烈推薦 Axios。它內建了 JSON 自動轉換、請求超時設定、以及強大的攔截器功能,能大幅減少重複程式碼。
Q2:為什麼 Fetch 在 HTTP 404 時不會跳到 .catch?
A:這是 Fetch 的設計特性。它只有在「網路中斷」或「請求被阻擋」時才會拒絕 (Reject)。如果是 HTTP 404 或 500,Fetch 仍視為成功。您必須手動檢查 response.ok 屬性來處理錯誤。
Q3:如何處理跨網域 (CORS) 的 AJAX 請求?
A:CORS 主要是伺服器端的設定。伺服器必須在回應標頭中包含 Access-Control-Allow-Origin。若前端需要攜帶 Cookie,請確保後端設定允許並在 AJAX 請求中開啟 withCredentials (XHR) 或 credentials: 'include' (Fetch)。
喜歡這篇文章,請幫忙拍拍手喔 🤣
