寫在前面:一般 SPA 中前端路由有2種實現方式:html
下面就來介紹下這兩種方式具體怎麼實現的前端
一.historyvue
1.history基本介紹vue-router
window.history 對象包含瀏覽器的歷史,window.history 對象在編寫時可不使用 window 這個前綴。history是實現SPA前端路由是一種主流方法,它有幾個原始方法:後端
在HTML5,history對象提出了 pushState() 方法和 replaceState() 方法,這兩個方法能夠用來向歷史棧中添加數據,就好像 url 變化了同樣(過去只有 url 變化歷史棧纔會變化),這樣就能夠很好的模擬瀏覽歷史和前進後退了,如今的前端路由也是基於這個原理實現的。api
2.history.pushState數組
pushState(stateObj, title, url) 方法向歷史棧中寫入數據,其第一個參數是要寫入的數據對象(不大於640kB),第二個參數是頁面的 title, 第三個參數是 url (相對路徑)。瀏覽器
關於pushState,有幾個值得注意的地方:服務器
pushState方法不會觸發頁面刷新,只是致使history對象發生變化,地址欄會有反應,只有當觸發前進後退等事件(back()和forward()等)時瀏覽器纔會刷新網絡
這裏的 url 是受到同源策略限制的,防止惡意腳本模仿其餘網站 url 用來欺騙用戶,因此當違背同源策略時將會報錯
3.history.replaceState
replaceState(stateObj, title, url) 和pushState的區別就在於它不是寫入而是替換修改瀏覽歷史中當前紀錄,其他和 pushState如出一轍
4.popstate事件
定義:每當同一個文檔的瀏覽歷史(即history對象)出現變化時,就會觸發popstate事件。
注意:僅僅調用pushState方法或replaceState方法 ,並不會觸發該事件,只有用戶點擊瀏覽器倒退按鈕和前進按鈕,或者使用JavaScript調用back、forward、go方法時纔會觸發。另外,該事件只針對同一個文檔,若是瀏覽歷史的切換,致使加載不一樣的文檔,該事件也不會觸發。
用法:使用的時候,能夠爲popstate事件指定回調函數。這個回調函數的參數是一個event事件對象,它的state屬性指向pushState和replaceState方法爲當前URL所提供的狀態對象(即這兩個方法的第一個參數)。
5.history實現spa前端路由代碼
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
<a class=
"api a"
>a.html</a>
<a class=
"api b"
>b.html</a>
// 註冊路由
document.querySelectorAll(
'.api'
).forEach(item => {
item.addEventListener(
'click'
, e => {
e.preventDefault();
let link = item.textContent;
if
(!!(window.history && history.pushState)) {
// 支持History API
window.history.pushState({name:
'api'
}, link, link);
}
else
{
// 不支持,可以使用一些Polyfill庫來實現
}
},
false
)
});
// 監聽路由
window.addEventListener(
'popstate'
, e => {
console.log({
location: location.href,
state: e.state
})
},
false
)
|
popstate監聽函數裏打印的e.state即是history.pushState()裏傳入的第一個參數,在這裏即爲{name: 'api'}
二.Hash
1.Hash基本介紹
url 中能夠帶有一個 hash http://localhost:9000/#/a.html
window 對象中有一個事件是 onhashchange,如下幾種狀況都會觸發這個事件:
2.Hash實現spa前端路由代碼
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
// 註冊路由
document.querySelectorAll(
'.api'
).forEach(item => {
item.addEventListener(
'click'
, e => {
e.preventDefault();
let link = item.textContent;
location.hash = link;
},
false
)
});
// 監聽路由
window.addEventListener(
'hashchange'
, e => {
console.log({
location: location.href,
hash: location.hash
})
},
false
)
|
hash模式與history模式,這兩種模式都是經過瀏覽器接口實現的,除此以外vue-router還爲非瀏覽器環境準備了一個abstract模式,其原理爲用一個數組stack模擬出瀏覽器歷史記錄棧的功能。固然,以上只是一些核心邏輯,爲保證系統的魯棒性源碼中還有大量的輔助邏輯,也很值得學習。
3、兩種模式比較
4、history模式的一個問題
咱們知道對於單頁應用來說,理想的使用場景是僅在進入應用時加載index.html,後續在的網絡操做經過Ajax完成,不會根據URL從新請求頁面,可是不免遇到特殊狀況,好比用戶直接在地址欄中輸入並回車,瀏覽器重啓從新加載應用等。
hash模式僅改變hash部分的內容,而hash部分是不會包含在HTTP請求中的:
http://oursite.com/#/user/id // 如從新請求只會發送http://oursite.com/
故在hash模式下遇到根據URL請求頁面的狀況不會有問題。
而history模式則會將URL修改得就和正常請求後端的URL同樣
http://oursite.com/user/id
在此狀況下從新向後端發送請求,如後端沒有配置對應/user/id的路由處理,則會返回404錯誤。
官方推薦的解決辦法是在服務端增長一個覆蓋全部狀況的候選資源:若是 URL 匹配不到任何靜態資源,則應該返回同一個 index.html 頁面,這個頁面就是你 app 依賴的頁面。同時這麼作之後,服務器就再也不返回 404 錯誤頁面,由於對於全部路徑都會返回 index.html 文件。爲了不這種狀況,在 Vue 應用裏面覆蓋全部的路由狀況,而後在給出一個 404 頁面。或者,若是是用 Node.js 做後臺,可使用服務端的路由來匹配 URL,當沒有匹配到路由的時候返回 404,從而實現 fallback。
參考資料:
一、瀏覽器History API :https://developer.mozilla.org/zh-CN/docs/Web/API/History_API
二、解決History模式訪問404的方案:https://router.vuejs.org/zh/guide/essentials/history-mode.html#%E5%90%8E%E7%AB%AF%E9%85%8D%E7%BD%AE%E4%BE%8B%E5%AD%90