如何用javascript实现路由功能【教程】
发布时间 - 2026-01-30 00:00:00 点击率:次浏览器原生 history.pushState 和 hashchange 可实现轻量前端路由:hash 模式用 hashchange(兼容 IE8+,URL 带 #),history 模式用 popstate(需服务端 fallback,URL 干净);首次加载需手动匹配路由;pushState 的 state 传状态对象、title 传空字符串、url 传同源相对路径;路由匹配建议用 path-to-regexp 等库处理动态路径;所有跳转(含手动输入 URL)均需服务端将前端路由 fallback 至 index.html。
浏览器原生支持的 history.pushState 和 hashchange 事件已足够实现轻量级前端路由,无需框架也能跑通核心逻辑。
监听 URL 变化用 popstate 还是 hashchange?
取决于你选的是 histo

-
hashchange仅监听 URL 中#后面的变化,兼容性极好(IE8+),但 URL 会带#/user这种不美观的前缀 -
popstate监听pushState/replaceState触发的前进后退,支持干净 URL(如/user),但需要服务端配合——否则直接刷新会 404 - 单页应用首次加载时,
popstate不会触发,必须手动调用一次路由匹配逻辑
pushState 的三个参数到底怎么填?
调用形式是 history.pushState(state, title, url),其中:
-
state:任意可序列化的 JS 值(如{page: 'user', id: 123}),会在popstate事件中作为event.state返回,用于恢复页面状态 -
title:目前所有主流浏览器都忽略该参数,传空字符串''即可 -
url:要推入历史栈的相对路径(如'/post/42'),浏览器地址栏会更新,但不会发起真实请求
注意:url 必须同源,且不能跨域或写成绝对 URL(如 https://example.com/a)——否则抛出安全错误。
如何匹配路由并渲染对应组件?
最简做法是维护一个路由表,用正则或 startsWith 判断当前路径:
const routes = [
{ path: '/', component: HomePage },
{ path: '/user', component: UserList },
{ path: '/user/:id', component: UserDetail }
];
function matchRoute(pathname) {
for (const route of routes) {
// 简单前缀匹配(适合无嵌套路由场景)
if (pathname.startsWith(route.path.replace(/\/:([^/]+)/g, '/[^/]+'))) {
return route;
}
}
return null;
}
// 手动触发匹配(首次加载 + popstate 时调用)
function navigate() {
const current = location.pathname;
const matched = matchRoute(current);
if (matched) render(matched.component);
}
实际项目中建议用更健壮的路径解析库(如 path-to-regexp),否则带参数的动态路由(/user/:id)容易漏匹配或误匹配。
点击链接时不刷新页面的关键在哪?
必须阻止 标签默认行为,并手动调用 pushState:
- 给所有路由链接加
data-nav属性(避免劫持外部链接) - 用事件委托监听
click,拿到e.target.href后截取 pathname - 调用
history.pushState更新地址栏,再立即执行navigate() - 别忘了处理右键、Ctrl+点击等非主键点击,这些不应拦截
容易被忽略的是:SPA 中所有跳转都得走 JS 路由逻辑,包括用户手动输入 URL 后回车——这意味着服务端必须把所有前端路由路径都 fallback 到 index.html,否则直接访问 /admin 会 404。
# javascript
# java
# html
# js
# 前端
# 浏览器
# 栈
# ai
# 路由
# 跨域
# gate
# 字符串
# 委托
# Event
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
java ZXing生成二维码及条码实例分享
黑客入侵网站服务器的常见手法有哪些?
Laravel如何生成PDF或Excel文件_Laravel文档导出工具与使用教程
Laravel如何生成API文档?(Swagger/OpenAPI教程)
Windows10如何删除恢复分区_Win10 Diskpart命令强制删除分区
制作ppt免费网站有哪些,有哪些比较好的ppt模板下载网站?
使用豆包 AI 辅助进行简单网页 HTML 结构设计
Android滚轮选择时间控件使用详解
如何快速上传自定义模板至建站之星?
昵图网官方站入口 昵图网素材图库官网入口
使用C语言编写圣诞表白程序
Laravel观察者模式如何使用_Laravel Model Observer配置
Laravel怎么实现模型属性的自动加密
如何快速启动建站代理加盟业务?
进行网站优化必须要坚持的四大原则
Laravel路由Route怎么设置_Laravel基础路由定义与参数传递规则【详解】
JS中对数组元素进行增删改移的方法总结
怎么用AI帮你设计一套个性化的手机App图标?
Laravel怎么判断请求类型_Laravel Request isMethod用法
详解jQuery中基本的动画方法
手机钓鱼网站怎么制作视频,怎样拦截钓鱼网站。怎么办?
Laravel Docker环境搭建教程_Laravel Sail使用指南
深圳网站制作设计招聘,关于服装设计的流行趋势,哪里的资料比较全面?
如何获取上海专业网站定制建站电话?
Laravel如何实现多对多模型关联?(Eloquent教程)
Laravel怎么配置自定义表前缀_Laravel数据库迁移与Eloquent表名映射【步骤】
利用 Google AI 进行 YouTube 视频 SEO 描述优化
linux top下的 minerd 木马清除方法
php json中文编码为null的解决办法
微信小程序 闭包写法详细介绍
如何快速建站并高效导出源代码?
如何在云指建站中生成FTP站点?
Microsoft Edge如何解决网页加载问题 Edge浏览器加载问题修复
如何登录建站主机?访问步骤全解析
湖南网站制作公司,湖南上善若水科技有限公司做什么的?
Laravel如何操作JSON类型的数据库字段?(Eloquent示例)
JavaScript数据类型有哪些_如何准确判断一个变量的类型
如何选择可靠的免备案建站服务器?
Python结构化数据采集_字段抽取解析【教程】
Android使用GridView实现日历的简单功能
Laravel如何创建自定义Artisan命令?(代码示例)
JS弹性运动实现方法分析
Laravel如何实现邮件验证激活账户_Laravel内置MustVerifyEmail接口配置【步骤】
php增删改查怎么学_零基础入门php数据库操作必知基础【教程】
JavaScript如何实现继承_有哪些常用方法
canvas 画布在主流浏览器中的尺寸限制详细介绍
如何快速搭建虚拟主机网站?新手必看指南
如何在阿里云部署织梦网站?
香港服务器网站搭建教程-电商部署、配置优化与安全稳定指南
用v-html解决Vue.js渲染中html标签不被解析的问题

