赞
踩
你越是认真生活,你的生活就会越美好
——弗兰克·劳埃德·莱特
《人生果实》经典语录
前端路由有两种模式:hash 模式和 history 模式,接下来分析这两种模式的实现方式和优缺点。
为了方便演示,需全局安装 light-server
:
yarn global add light-server
// 或者
npm -g install light-server
hash | history | |
url显示 | 有#,很Low | 无#,好看 |
回车刷新 | 可以加载到hash值对应页面 | 一般就是404掉了 |
支持版本 | 支持低版本浏览器和IE浏览器 | HTML5新推出的API |
hash 模式
是一种把前端路由的路径用井号 #
拼接在真实 URL 后面的模式。
当井号 #
后面的路径发生变化时,浏览器并不会重新发起请求
,而是会触发 hashchange 事件
。
我们新建一个 hash.html
文件,内容为:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>hash</title> </head> <body> <a href="#/a">A页面</a> <a href="#/b">B页面</a> <div id="app"></div> <script> function render() { const app = document.querySelector('#app') app.innerHTML = window.location.hash } window.addEventListener('hashchange', render) </script> </body> </html>
在目录下运行:
light-server -s . --port 3000
然后打开 http://localhost:3000/hash.html
查看效果。
在上面的例子中,我们利用 a 标签
设置了两个路由导航,把 app 当做视图渲染容器
,当切换路由时
触发视图容器更新
,这其实就是大多数前端框架哈希路由
的实现原理。
优点
缺点
history API 是 H5 提供的新特性,允许开发者直接更改前端路由,即更新浏览器 URL 地址
而不重新发起请求
(将url替换并且不刷新页面)。
我们新建一个 history.html
,内容为:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>history</title> </head> <body> <a href="javascript:toA();">A页面</a> <a href="javascript:toB();">B页面</a> <div id="app"></div> <script> function render() { console.log('render') app.innerHTML = window.location.pathname } function toA() { history.pushState({}, null, '/a') render() } function toB() { history.pushState({}, null, '/b') render() } window.addEventListener('popstate', render) </script> </body> </html>
在目录下运行:
light-server -s . --historyindex '/history.html' --port 3000
然后打开 http://localhost:3000/history.html
查看效果。
history API 提供了丰富的函数供开发者调用,我们不妨把控制台
打开,然后输入下面的语句来观察浏览器地址栏的变化:
history.replaceState({}, null, '/b') // 替换路由
history.pushState({}, null, '/a') // 路由压栈 替换当前地址 被替换地址进入访问历史
history.back() // 返回
history.forward() // 前进
history.go(-2) // 后退2次
上面的代码监听了 popstate 事件
,该事件能监听到
:
前进和后退
操作back
、forward
和 go
方法监听不到
pushState
和 replaceState
方法这也是为什么上面的 toA 和 toB 函数内部需要手动调用 render 方法的原因。
另外,大家可能也注意到 light-server 的命令多了 --historyindex '/history.html'
参数,这是干什么的呢?
浏览器在刷新的时候,会按照路径
发送真实的资源请求,如果这个路径是前端通过 history API 设置的 URL,那么在服务端往往不存在这个资源,于是就返回 404
了。
上面的参数的意思就是如果后端资源不存在就返回 history.html 的内容
。(试了没反应,知道有这种用法即可)
因此在线上部署
基于 history API
的单页面应用
的时候,一定要后端配合
支持才行,否则会出现大量的 404
。
以最常用的 Nginx 为例,只需要在配置的 location / 中增加下面一行即可:
try_files $uri /index.html
优点:
缺点:
传统的路由指的是:当用户访问一个url时,对应的服务器会接收这个请求,然后解析url中的路径,从而执行对应的处理逻辑。这样就完成了一次路由分发。
而前端路由不涉及服务器,是前端利用hash或者HTML5的history API来实现的,一般用于不同内容的展示和切换。
谢谢你阅读到了最后~
期待你关注、收藏、评论、点赞~
让我们一起 变得更强
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。