从理解路由到实现一套Router(路由)

小伙伴们大家好啊,我是李牌牌。平时在Vue项目中经常用到路由,但是也仅仅处于会用的层面,很多基础知识并不是真正的理解。于是牌牌呢查阅了很多资料,总结下路由相关的知识,查缺不漏,加深自己对路由的理解。

路由

在 Web 开发过程中,经常遇到路由的概念。那么到底什么是路由呢?简单来说,路由就是 URL 到函数的映射。

路由这个概念本来是由后端提出来的,在以前用模板引擎开发页面的时候,是使用路由返回不同的页面,大致流程是这样的:

  1. 浏览器发出请求;

  2. 服务器监听到 80 或者 443 端口有请求过来,并解析 UR L路径;

  3. 服务端根据路由设置,查询相应的资源,可能是 html 文件,也可能是图片资源......,然后将这些资源处理并返回给浏览器;

  4. 浏览器接收到数据,通过content-type决定如何解析数据

简单来说,路由就是用来跟后端服务器交互的一种方式,通过不同的路径来请求不同的资源,请求HTML页面只是路由的其中一项功能。

服务端路由

当服务端接收到客户端发来的 HTTP 请求时,会根据请求的 URL,找到相应的映射函数,然后执行该函数,并将函数的返回值发送给客户端。

对于最简单的静态资源服务器,可以认为,所有 URL 的映射函数就是一个文件读取操作。对于动态资源,映射函数可能是一个数据库读取操作,也可能进行一些数据处理,等等。

客户端路由

服务端路由会造成服务器压力比较大,而且用户访问速度也比较慢。在这种情况下,出现了单页应用。

单页应用,就是只有一个页面,用户访问网址,服务器返回的页面始终只有一个,不管用户改变了浏览器地址栏的内容或者在页面发生了跳转,服务器不会重新返回新的页面,而是通过相应的js操作来实现页面的更改。

前端路由其实就是:通过地址栏内容的改变,显示不同的页面

前端路由的优点:

  • 前端路由可以让前端自己维护路由与页面展示的逻辑,每次页面改动不需要通知服务端。

  • 更好的交互体验:不用每次从服务端拉取资源。

前端路由的缺点: 使用浏览器的前进、后退键时会重新发送请求,来获取数据,没有合理利用缓存。

前端路由实现原理: 本质就是监测 URL 的变化,通过拦截 URL 然后解析匹配路由规则。

前端路由的实现方式

  1. hash模式(location.hash + hashchange 事件)

hash 模式的实现方式就是通过监听 URL 中的 hash 部分的变化,触发haschange事件,页面做出不同的响应。但是 hash 模式下,URL 中会带有 #,不太美观。

  1. history模式

history 路由模式的实现,基于 HTML5 提供的 History 全局对象,它的方法有:

  • history.go():在会话历史中向前或者向后移动指定页数

  • history.forward():在会话历史中向前移动一页,跟浏览器的前进按钮功能相同

  • history.back():在会话历史记录中向后移动一页,跟浏览器的后腿按钮功能相同

  • history.pushState():向当前浏览器会话的历史堆栈中添加一个状态,会改变当前页面url,但是不会伴随这刷新

  • history.replaceState():将当前的会话页面的url替换成指定的数据,replaceState 会改变当前页面的url,但也不会刷新页面

  • window.onpopstate:当前活动历史记录条目更改时,将触发popstate事件

history路由的实现,主要是依靠pushStatereplaceStatewindow.onpopstate实现的。但是有几点要注意:

  • 当活动历史记录条目更改时,将触发 popstate 事件;

  • 调用history.pushState()history.replaceState()不会触发 popstate 事件

  • popstate 事件只会在浏览器某些行为下触发,比如:点击后退、前进按钮(或者在 JavaScript 中调用history.back()history.forward()history.go()方法)

  • a 标签的锚点也会触发该事件

对 pushState 和 replaceState 行为的监听

如果想监听 pushState 和 replaceState 行为,可以通过在方法里面主动去触发 popstate 事件,另一种是重写history.pushState,通过创建自己的eventedPushState自定义事件,并手动派发,实际使用过程中就可以监听了。具体做法如下:

function eventedPushState(state, title, url) {var pushChangeEvent = new CustomEvent("onpushstate", {detail: {state,title,url}});document.dispatchEvent(pushChangeEvent);return history.pushState(state, title, url);
}document.addEventListener("onpushstate",function(event) {console.log(event.detail);},false
);eventedPushState({}, "", "new-slug"); 

router 和 route 的区别

route 就是一条路由,它将一个 URL 路径和一个函数进行映射。而 router 可以理解为一个容器,或者说一种机制,它管理了一组 route。

概括为:route 只是进行了 URL 和函数的映射,在当接收到一个 URL 后,需要去路由映射表中查找相应的函数,这个过程是由 router 来处理的。

动态路由和静态路由

  • 静态路由

静态路由只支持基于地址的全匹配。

  • 动态路由

动态路由除了可以兼容全匹配外还支持多种”高级匹配模式“,它的路径地址中含有路径参数,使得它可以按照给定的匹配模式将符合条件的一个或多个地址映射到同一个组件上。

动态路由一般结合角色权限控制使用。

动态路由的存储有两种方式:

  1. 将路由存储到前端

  2. 将路由存储到数据库

动态路由的好处:

  • 灵活,无需手动维护

  • 存储到数据库,增加安全性

实现一个路由

一个简单的Router应该具备哪些功能
  • 以 Vue为例,需要有 <router-link>链接、<router-view>容器、component组件和path路由路径:

<div id="app"><h1>Hello World</h1><p><!-- 使用 router-link 组件进行导航 --><!-- 通过传递 to 来指定链接 --><!-- <router-link> 将呈现一个带有正确 href属性的<a>标签 --><router-link to="/">Go to Home</router-link><router-link to="/about">Go to About</router-link></p><!-- 路由出口 --><!-- 路由匹配到的组件将渲染在这里 --><router-view></router-view>
</div>
const routes = [{path: '/',component: Home
},
{path: '/about',component: About
}]
  • 以React为例,需要有<BrowserRouter>容器、<Route>路由、组件和链接:

<BrowserRouter><Routes><Route path="/" element={<App />}><Route index element={<Home />} /><Route path="teams" element={<Teams />}><Route path=":teamId" element={<Team />} /><Route path="new" element={<NewTeamForm />} /><Route index element={<LeagueStandings />} /></Route></Route></Routes>
</BrowserRouter>
<div><h1>Home</h1><nav><Link to="/">Home</Link> | {""}<Link to="about">About</Link></nav>
</div>
  • 综上,一个简单的 Router 应该具备以下功能:

    • 容器(组件)

    • 路由

    • 业务组件 & 链接组件

不借助第三方工具库,如何实现路由

不借助第三方工具库实现路由,我们需要思考以下几个问题:

  • 如何实现自定义标签,如vue的<router-view>,React的<Router>

  • 如何实现业务组件

  • 如何动态切换路由

准备工作
1、根据对前端路由 history 模式的理解,将大致过程用如下流程图表示:

图片


2、如果不借助第三方库,我们选择使用 Web components 。Web Components由三项主要技术组成,它们可以一起使用来创建封装功能的定制元素。

    • Custom elements(自定义元素) :一组JavaScript API,允许我们定义 custom elements及其行为,然后可以在界面按照需要使用它们。

    • Shadow DOM(影子DOM) :一组JavaScript API,用于将封装的“影子”DOM树附加到元素(与主文档分开呈现)并控制关联的功能。通过这种方式,可以保持元素的功能私有。

    • HTML template(HTML模版) :<template><slot>可以编写不在页面显示的标记模板,然后它们可以作为自定义元素结构的基础被多次重用。

另外还需要注意 Web Components 的生命周期:

connectedCallback:当 custom element 首次被插入文档DOM时,被调用

disconnectedCallback:当 custom element 从文档DOM中删除时,被调用

adoptedCallback:当custom element 被移动到新的文档时,被调用

attributeChangedCallback:当 custom element 增加、删除、修改自身属性时,被调用

3、Shadow DOM

    • open:shadow root 元素可以从 js 外部访问根节点

    • close :拒绝从 js 外部访问关闭的 shadow root 节点

    • 语法:const shadow = this.attachShadow({mode:closed});

    • Shadow host:一个常规DOM节点,Shadow DOM 会被附加到这个节点上

    • Shadow tree:Shadow DOM 内部的 DOM 树

    • Shadow boundary:Shadow DOM 结束的地方,也是常规DOM开始的地方

    • Shadow root:Shadow tree 的根节点

    • Shadow DOM 特有的术语:

    • Shadow DOM的重要参数mode:

  1. 通过自定义标签创建容器组件、路由、业务组件和链接组件标签,使用

CustomElementRegistry.define()注册自定义元素。其中,Custom elements 的简单写法举例:

<my-text></my-text><script>class MyText extends HTMLElement{constructor(){super();this.append(“我的文本”);}}window.customElements.define("my-text",MyText);
</script>
  1. 组件的实现可以使用 Web Components,但是这样有缺点,我们没有打包引擎处理 Web Components组件,将其全部加载过来。

为了解决以上问题,我们选择动态加载,远程去加载一个 html 文件。html文件里面的结构如下:支持模版(template),脚本(template),脚本(script),样式(style),非常地像vue。组件开发模版如下:

<template><div>商品详情</div><div id="detail">商品ID:<span id="product-id" class="product-id"></span></div>
</template><script>this.querySelector("#product-id").textContent = history.state.id;
</script><style>.product-id{color:red;}
</style>
  1. 监听路由的变化:

popstate可以监听大部分路由变化的场景,除了pushState 和 replaceState

pushState 和 replaceState可以改变路由,改变历史记录,但是不能触发popstate事件,需要自定义事件并手动触发自定义事件,做出响应。

  1. 整体架构图如下:

图片

8.  组件功能拆解分析如下:

  • 链接组件 — CustomLink(c-link)

当用户点击<c-link>标签后,通过event.preventDefault();阻止页面默认跳转。根据当前标签的to属性获取路由,通过history.pushState("","",to)进行路由切换。

//  <c-link to="/" class="c-link">首页</c-link>
class CustomLink extends HTMLElement {connectedCallback() {this.addEventListener("click", ev => {ev.preventDefault();const to = this.getAttribute("to");// 更新浏览器历史记录history.pushState("", "", to)})}
}
window.customElements.define("c-link", CustomLink);
  • 容器组件 — CustomRouter(c-router)

主要是收集路由信息,监听路由信息的变化,然后加载对应的组件

  • 路由 — CustomRoute(c-route)

主要是提供配置信息,对外提供getData 的方法

// 优先于c-router注册
//  <c-route path="/" component="home" default></c-route>
class CustomRoute extends HTMLElement {#data = null;getData() {return {default: this.hasAttribute("default"),path: this.getAttribute("path"),component: this.getAttribute("component")}}
}
window.customElements.define("c-route", CustomRoute);
  • 业务组件 — CustomComponent(c-component)

实现组件,动态加载远程的html,并解析

完整代码实现

index.html:

<div class="product-item">测试的产品</div>
<div class="flex"><ul class="menu-x"><c-link to="/" class="c-link">首页</c-link><c-link to="/about" class="c-link">关于</c-link></ul>
</div>
<div><c-router><c-route path="/" component="home" default></c-route><c-route path="/detail/:id" component="detail"></c-route><c-route path="/about" component="about"></c-route></c-router>
</div><script src="./router.js"></script>

home.html:

<template><div>商品清单</div><div id="product-list"><div><a data-id="10" class="product-item c-link">香蕉</a></div><div><a data-id="11" class="product-item c-link">苹果</a></div><div><a data-id="12" class="product-item c-link">葡萄</a></div></div>
</template><script>let container = this.querySelector("#product-list");// 触发历史更新// 事件代理container.addEventListener("click", function (ev) {console.log("item clicked");if (ev.target.classList.contains("product-item")) {const id = +ev.target.dataset.id;history.pushState({id}, "", `/detail/${id}`)}})
</script><style>.product-item {cursor: pointer;color: blue;}
</style>

detail.html:

<template><div>商品详情</div><div id="detail">商品ID:<span id="product-id" class="product-id"></span></div>
</template><script>this.querySelector("#product-id").textContent=history.state.id;
</script><style>.product-id{color:red;}
</style>

about.html:

<template>About Me!
</template>

route.js:

const oriPushState = history.pushState;// 重写pushState
history.pushState = function (state, title, url) {// 触发原事件oriPushState.apply(history, [state, title, url]);// 自定义事件var event = new CustomEvent("c-popstate", {detail: {state,title,url}});window.dispatchEvent(event);
}// <c-link to="/" class="c-link">首页</c-link>
class CustomLink extends HTMLElement {connectedCallback() {this.addEventListener("click", ev => {ev.preventDefault();const to = this.getAttribute("to");// 更新浏览历史记录history.pushState("", "", to);})}
}
window.customElements.define("c-link", CustomLink);// 优先于c-router注册
// <c-toute path="/" component="home" default></c-toute>
class CustomRoute extends HTMLElement {#data = null;getData() {return {default: this.hasAttribute("default"),path: this.getAttribute("path"),component: this.getAttribute("component")}}
}
window.customElements.define("c-route", CustomRoute);// 容器组件
class CustomComponent extends HTMLElement {async connectedCallback() {console.log("c-component connected");// 获取组件的path,即html的路径const strPath = this.getAttribute("path");// 加载htmlconst cInfos = await loadComponent(strPath);const shadow = this.attachShadow({ mode: "closed" });// 添加html对应的内容this.#addElement(shadow, cInfos);}#addElement(shadow, info) {// 添加模板内容if (info.template) {shadow.appendChild(info.template.content.cloneNode(true));}// 添加脚本if (info.script) {// 防止全局污染,并获得根节点var fun = new Function(`${info.script.textContent}`);// 绑定脚本的this为当前的影子根节点fun.bind(shadow)();}// 添加样式if (info.style) {shadow.appendChild(info.style);}}
}
window.customElements.define("c-component", CustomComponent);// <c-router></c-router>
class CustomRouter extends HTMLElement {#routesconnectedCallback() {const routeNodes = this.querySelectorAll("c-route");console.log("routes:", routeNodes);// 获取子节点的路由信息this.#routes = Array.from(routeNodes).map(node => node.getData());// 查找默认的路由const defaultRoute = this.#routes.find(r => r.default) || this.#routes[0];// 渲染对应的路由this.#onRenderRoute(defaultRoute);// 监听路由变化this.#listenerHistory();}// 渲染路由对应的内容#onRenderRoute(route) {var el = document.createElement("c-component");el.setAttribute("path", `/${route.component}.html`);el.id = "_route_";this.append(el);}// 卸载路由清理工作#onUploadRoute(route) {this.removeChild(this.querySelector("#_route_"));}// 监听路由变化#listenerHistory() {// 导航的路由切换window.addEventListener("popstate", ev => {console.log("onpopstate:", ev);const url = location.pathname.endsWith(".html") ? "/" : location.pathname;const route = this.#getRoute(this.#routes, url);this.#onUploadRoute();this.#onRenderRoute(route);});// pushStat或replaceSatewindow.addEventListener("c-popstate", ev => {console.log("c-popstate:", ev);const detail = ev.detail;const route = this.#getRoute(this.#routes, detail.url);this.#onUploadRoute();this.#onRenderRoute(route);})}// 路由查找#getRoute(routes, url) {return routes.find(function (r) {const path = r.path;const strPaths = path.split('/');const strUrlPaths = url.split("/");let match = true;for (let i = 0; i < strPaths.length; i++) {if (strPaths[i].startsWith(":")) {continue;}match = strPaths[i] === strUrlPaths[i];if (!match) {break;}}return match;})}
}
window.customElements.define("c-router", CustomRouter);// 动态加载组件并解析
async function loadComponent(path, name) {this.caches = this.caches || {};// 缓存存在,直接返回if (!!this.caches[path]) {return this.caches[path];}const res = await fetch(path).then(res => res.text());// 利用DOMParser校验const parser = new DOMParser();const doc = parser.parseFromString(res, "text/html");// 解析模板,脚本,样式const template = doc.querySelector("template");const script = doc.querySelector("script");const style = doc.querySelector("style");// 缓存内容this.caches[path] = {template,script,style}return this.caches[path];
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/460452.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MFC图形函数学习04——画矩形函数

MFC中绘制矩形函数是MFC的基本绘图函数&#xff0c;它的大小和位置由左上角和右下角的坐标决定&#xff1b;若想绘制的矩形边框线型、线宽、颜色以及填充颜色都还需要其它函数的配合。 一、绘制矩形函数 原型&#xff1a;BOOL Rectangle(int x1,int y1,int x2,int y2); …

Kafka 与传统 MQ 消息系统之间有三个关键区别?

大家好&#xff0c;我是锋哥。今天分享关于【Kafka 与传统 MQ 消息系统之间有三个关键区别&#xff1f;】面试题&#xff1f;希望对大家有帮助&#xff1b; Kafka 与传统 MQ 消息系统之间有三个关键区别&#xff1f; 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 …

TLKS-PMG-100BM这款输电线路智能多目视频监控装置,它具体有哪些亮点和优势?

TLKS-PMG-100BM输电线路智能多目视频监控装置&#xff08;输电线路全景视频监控装置、输电线路云台变焦视频监控装置&#xff09;无疑是一款功能全面、性能卓越的输电线路智能监控装置。它配备了水平360、垂直90旋转的全向云台摄像头&#xff0c;能够轻松实现全景视野监视&…

Java中的运算符【与C语言的区别】

目录 1. 算术运算符 1.0 赋值运算符&#xff1a; 1.1 四则运算符&#xff1a; - * / % 【取余与C有点不同】 1.2 增量运算符&#xff1a; - * / % * 【右侧运算结果会自动转换类型】 1.3 自增、自减&#xff1a;、-- 2. 关系运算符 3. 逻辑运算符 3.1 短路求值 3.2 【…

目标检测:YOLOv11(Ultralytics)环境配置,适合0基础纯小白,超详细

目录 1.前言 2. 查看电脑状况 3. 安装所需软件 3.1 Anaconda3安装 3.2 Pycharm安装 4. 安装环境 4.1 安装cuda及cudnn 4.1.1 下载及安装cuda 4.1.2 cudnn安装 4.2 创建虚拟环境 4.3 安装GPU版本 4.3.1 安装pytorch&#xff08;GPU版&#xff09; 4.3.2 安装ultral…

HT7178 带输出关断的20V,14A全集成同步升压转换器

1、特点 输入电压范围VpIN:2.7V-20V 输出电压范围VouT:4.5V-20V 可编程峰值电流:14A 高转换效率: 95%(VPIN7.2V, VoUT 16V, IouT3A) 94%(VPIN12V,VoUT18V,IoUT4A) 90%(VPIN3.3, VoUT-9V,IOUT3A) 轻载条件下两种调制方式:脉频调制(PFM)和 强制脉宽调试(PWM) 集成输出关断的栅极…

使用axios请求分页

npm install axios <template><div><el-table :data"items" style"width: 100%"><el-table-column prop"id" label"ID" /><el-table-column prop"name" label"名称" /><!-- 添…

基于SpringBoot的在线医疗问答平台

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、Vue项目源码、SSM项目源码、微信小程序源码 精品专栏&#xff1a;…

codeforces _ 补题

C. Ball in Berland 传送门&#xff1a;Problem - C - Codeforces 题意&#xff1a; 思路&#xff1a;容斥原理 考虑 第 i 对情侣组合 &#xff0c;男生为 a &#xff0c;女生为 b &#xff0c;那么考虑与之匹配的情侣 必须没有 a | b &#xff0c;一共有 k 对情侣&#x…

【Canvas与图标】长方形牛皮纸文件袋图标

【成图】 120*120的图标 大图 小图&#xff1a; 【代码】 <!DOCTYPE html> <html lang"utf-8"> <meta http-equiv"Content-Type" content"text/html; charsetutf-8"/> <head><title>长方文件袋图标</title>…

奔走相告! ClickHouse 全新构建了强大的 JSON 数据类型

本文字数&#xff1a;8969&#xff1b;估计阅读时间&#xff1a;23 分钟 作者&#xff1a;Pavel Kruglov 本文在公众号【ClickHouseInc】首发 简介 JSON 已成为现代数据系统中处理半结构化和非结构化数据的首选格式。无论是在日志记录和可观测性 (observability) 应用场景、实…

统信UOS下启动图形界面应用工具manager报错:No protocol specified的解决办法

☞ ░ 前往老猿Python博客 ░ https://blog.csdn.net/LaoYuanPython 一、问题情况 达梦提供了丰富的图形界面工具&#xff0c;包括&#xff1a;manager、monitor、dbca等&#xff0c;但在统信操作系统进入终端去启动manager时报错&#xff1a;No protocol specified。 咨询了达…

【CSS3】css开篇基础(6)

1.❤️❤️前言~&#x1f973;&#x1f389;&#x1f389;&#x1f389; Hello, Hello~ 亲爱的朋友们&#x1f44b;&#x1f44b;&#xff0c;这里是E绵绵呀✍️✍️。 如果你喜欢这篇文章&#xff0c;请别吝啬你的点赞❤️❤️和收藏&#x1f4d6;&#x1f4d6;。如果你对我的…

【设计模式系列】迭代器模式(七)

一、什么是迭代器模式 迭代器模式&#xff08;Iterator Pattern&#xff09;是一种行为型设计模式&#xff0c;它提供一种方法来顺序访问一个聚合对象中的各个元素&#xff0c;而不暴露其内部的表示。迭代器模式将集合的遍历过程封装在一个独立的迭代器对象中&#xff0c;这样…

Linux线程安全(二)条件变量实现线程同步

目录 条件变量 条件变量初始化和唤醒 键盘触发条件变量唤醒线程demo 条件变量的等待 条件变量定时等待demo 条线变量实现多线程间的同步 条件变量 条件变量是为了控制多个线程的同步工作而设计的 比如说一个系统中有多个线程的存在但有且仅有一个线程在工作&#xff0c…

数据结构---顺序表

文章目录 线性表顺序表的使用及其内部方法ArrayList 的扩容机制顺序表的几种遍历方式顺序表的优缺点顺序表的模拟实现杨辉三角扑克牌算法 线性表 线性表&#xff08;linear list&#xff09;是n个具有相同特性的数据元素的有限序列。 线性表是一种在实际中广泛使用的数据结构&a…

大模型,多模态大模型面试【LoRA,分类,动静态数据类型,DDPM,ControlNet,IP-Adapter, Stable Diffusion】

大模型&#xff0c;多模态大模型面试【LoRA&#xff0c;分类&#xff0c;动静态数据类型&#xff0c;DDPM&#xff0c;ControlNet&#xff0c;IP-Adapter, Stable Diffusion】 问题一&#xff1a;LoRA是用在节省资源的场景下&#xff0c;那么LoRA具体是节省了内存带宽还是显存呢…

【力扣刷题实战】相同的树

大家好&#xff0c;我是小卡皮巴拉 文章目录 目录 力扣题目&#xff1a; 相同的树 题目描述 示例 1&#xff1a; 示例 2&#xff1a; 示例 3&#xff1a; 解题思路 题目理解 算法选择 具体思路 解题要点 完整代码&#xff08;C语言&#xff09; 兄弟们共勉 &#…

2024年AI绘画与写作工具排行榜:提升创作效率必备利器推荐

2024年&#xff0c;AI绘画和写作工具迎来全新突破&#xff0c;从艺术创作到内容写作&#xff0c;越来越多的创作者开始依赖AI工具来提升效率、拓展创意。而随着市场上AI工具的增多&#xff0c;如何选择适合的工具变得尤为重要。今天为大家推荐一些提升创作效率的AI绘画和写作利…

沈阳乐晟睿浩科技有限公司抖音小店新篇章

在当今数字化时代&#xff0c;电商行业如雨后春笋般迅速崛起&#xff0c;其中抖音小店凭借其庞大的用户基础、精准的推荐算法和便捷的购物体验&#xff0c;成为了电商领域的新宠。在这场电商变革中&#xff0c;沈阳乐晟睿浩科技有限公司&#xff08;以下简称“乐晟睿浩”&#…