随着 ajax 的流行,异步数据请求体验极具提升,用户得以在不刷新浏览器的情况下进行页面交互,而异步交互体验的更高级版本就是 SPA —— 单页应用。
单页应用不仅仅是在页面交互时无刷新,连页面跳转都是无刷新的,为了实现单页应用,就有了前端路由。
常用的两种模式
类似于服务端路由解析对应的 url 路径,返回对应的页面/资源的方式,前端路由实现起来其实也很简单,就是匹配不同的 url 路径,进行解析,然后动态的渲染出区域 html 内容。
这样自然 url 每次变化的时候,都会造成页面的刷新。
那么在改变 url 的情况下,如何保证页面的不刷新?
hash 模式
在 2014 年之前,大家是通过 hash 来实现路由,url hash 就是类似于:
https://www.xxx.com/#/login
这种 # 后面 hash 值的变化,并不会导致浏览器向服务器发出请求,浏览器不发出请求,也就不会刷新页面。
为什么改变 hash 不刷新页面?——URL的井号‘#’
‘#’ 代表网页中的一个位置,它后面的字符,就是该位置的标识符,它只对浏览器有用,服务器不识别,因此 HTTP 请求不会包含 #
(想要请求 url 包含 # ,可使用 encodeURIComponent()
进行部分转义)
改变 hash ,只会让浏览器滚动到相应位置,不会重载网页
每次 hash 值的变化,会触发 hashchange 事件,通过window.onhashchange
监听该事件我们就可以检测变化的 hash 值来做相应的页面操作。
简易实现
接下来我们用最简单的代码实现 hash 模式,仅为了解其思想(你可以直接复制到一个 html 上并通过静态服务器如 http-server 查看):
Hash 路由
如何实现最基础的前进后退?
这里我们简单实现一下后退功能,前进思路类似:
Hash 路由
思路就是通过一个数组记录每次 hashchange 事件的 hash 值,点击后退时取出上一次 hash 值覆盖当前页面的 hash。
需要注意的是需要区别当前 hash 是后退生成(后退时的 hash 变化不应记录)的还是跳转生成,避免重复记录。
history 模式
可以看到,在早期 hash 模式虽然可以实现前端路由,但其后退前进操作就十分麻烦。
2014 年后,HTML5 引入了 History API,让我们能够快速访问页面历史。
其中 history.pushState() 和 history.replaceState() 方法,它们分别可以添加和修改历史记录条目,通过这两个 API 可以改变 url 地址而无须重新加载页面。
同时还有 popstate 事件:
通过window.onpopstate
可以监听在浏览器点击后退、前进按钮(或者在 JavaScript 中调用 history.back()、history.forward()、history.go() 方法) 触发的 popstate 事件。
通过这些就能用另一种方式来实现前端路由了,但原理都是跟 hash 实现相同的。
用 history 实现上面 hash 代码
History 路由
用了 HTML5 的实现,单页路由的 url 就不会多出一个 #,变得更加美观。
但因为没有 # 号,所以当用户刷新页面之类的操作时,浏览器还是会给服务器发送请求。
为了避免出现这种情况,history 模式需要服务器的支持,把所有路由都重定向到根页面。
如何监听 pushState 和 replaceState 的变化
经过理论及实践我们知道 replaceState(),pushState() 两个 API 不会触发 popstate 监听事件。
我们可以生成全新的 window 监听事件监听其变化:
function addListen(type) {
const source = history[type];
return function () {
const event = new Event(type);
event.arguments = arguments;
window.dispatchEvent(event);
return source.apply(this, arguments);
};
}
history.pushState = addListen("pushState");
history.replaceState = addListen("replaceState");
window.addEventListener("replaceState", (e) => {
console.log("我监听了 replaceState");
});
window.addEventListener("pushState", (e) => {
console.log("我监听了 pushState");
});
两种模式对比
- 无 # 的 history 模式更自然
- history 模式需要 IE9 以上,相对于 hash 模式的 IE8 兼容性差
- history 模式需服务器端配合,反过来说 hash 模式不支持服务端渲染
结语
以上就是前端路由的 hash 和 history 两种模式的主要原理及实现思路了,如果你觉得不错,别忘了点个赞!
本文参考:
面试官: 你了解前端路由吗?
阿里P7:你了解路由吗?
[实践系列]前端路由