vue中router路由的原理?两种路由模式如何实现?(vue2) -(上)

平时我们编写路由时,通常直接下载插件使用,在main.js文件中引入直接通过引入vue-router中的Router通过Vue.use使用以后定义一个routeMap数组,里边是我们编写路由的地方,最后通过实例化一个 Router实例 将routes=我们定义的routeMao路由数组。

但是我们并不知道它是如何实现的,因此我们可以通过自己编写插件的形式,实现一个vue-router

常用路由步骤:

//router/index.js
import Vue from 'vue'import Router from 'vue-router'Vue.use(Router)export const routeMap= [{path: '/home',component: () => import('../components/Home/Home.vue')},{path: '/list',component: () => import('../components/List/List.vue')},
]const router = new Router({routes: constantRouterMap
})export default router//main.js
import router from './router'new Vue({router,store,render: h => h(App),
}).$mount('#app')

我们来实现一下如何实现一个hash模式的路由

由于我们不适用插件形式实现,因此我们使用步骤与平常实现有一些差异。

router/index.js是我们的路由文件,里面放置我们的路由

plugin/router.js是我们的自定义插件文件,里面放置我们自定义插件内容

components是我们组件目录,放置我们的路由组件(通常使用view定义路由组件)

main.js是我们的入口文件,需要在该文件引入router并且实例化vue的时候需要将引入的router使用

1、首先我们在App.vue文件中,通过<router-view/>以及<router-link/>定义声明式导航和路由占位

<div><router-link to="/home">首页</router-link><router-link to="/list">列表</router-link><hr><router-view></router-view>
</div>

这里我们会发现使用以后会报错,因为我们没有下载插件,因此没有这两个组件。

我们需要配置plugin插件

由于没有这两个全局组件因此我们需要配置两个全局组件

install方法是为了将我们的路由挂载在我们的组件实例上,通过mixin全局混入,将我们的实例上挂载$router属性

然后定义两个全局组件,router-link由于它渲染出来相当于html中的a标签,使用render方法,参数为一个createElement方法接收三个参数(第一个参数创建的元素,第二个参数元素具有的属性, 第三个参数元素的内容)

router-view由于是一个路由占位符,因此返回的是个组件,渲染组件。

通过routesMap是我们存放的路由,current是我们的当前的path路由,因此可以通过查找路由中key值为我们当前path路径的,查找到我们的组件。通过return渲染我们的组件

VueRouter.install = function (_Vue) {//1、保存VueVue = _Vue//2、将以下代码延迟到vue实例初始化完毕执行Vue.mixin({beforeCreate() {//判断如果有router属性就执行下方代码if (this.$options.router) {Vue.prototype.$router = this.$options.router;}}})//创建全局组件router-link和router-view//创建router-link组件Vue.component('router-link', {props: {to: {type: String | Object,required: true}},render(h) {// h 有三个参数,第一个参数创建的元素,第二个参数元素具有的属性, 第三个参数元素的内容return h('a', { attrs: { href: '#' + this.to } }, this.$slots.default)}})//创建router-view组件Vue.component('router-view', {// 组件要重新渲染,必须要让数据发生改变的时候,重新去执行render函数render(h) {const router = this.$routerconsole.log(router.routesMap, 'routesMap');let component = router.routesMap[router.current];//组件//获取到外部传递的路由表return h(component)}})
}

2、我们定义一个路由类,该类会获取到我们的路由信息从而进行路由监听以及组件渲染。

(1)、该类中constructor是我们通过实例化vueRouter传入的数据,我们需要通过将数据保存在vueRouter实例上,获取到当前的url path路径。

(2)、通过监听方法hashchange监听我们hash值的变化,通过方法进行监听,但是一定要注意这里方法的this指向必须是vueRouter实例不可以是window,因此需要通过bind改变this指向

(3)、通过defineReactive来讲我们的path路径变为响应式,这样每次路径发生变化可以监测到变化。

(4)、定义routesMap为我们的路由对象,遍历我们传入的路由数组,将我们每个path路径对应组件,为一组一组的键值对。

这样可以实现一套hash路由模式的路由跳转。

全部代码如下:

//plugin/router.js
let Vue;class VueRouter {constructor(options) {//保存选项this.options = options;// console.log(options, '路由数据');// 定义一个响应式的变量current,保存当前的hash值let url = location.hash.slice(1,) || '/';// defineReactive定义响应式的对象Vue.util.defineReactive(this, 'current', url)// hashchange事件来监听hash值的变化, 注意this指向问题addEventListener('hashchange', this.changeHash.bind(this))this.routesMap = {};//对routes中的对象做一个映射:routesMap = {'/home':component,'/list': component}this.options.routes.forEach(route => {this.routesMap[route.path] = route.component})}//监听hash值变化的函数changeHash() {//通过location.hash来获取到hash值this.current = location.hash.slice(1,)console.log(this.routesMap, '555');// console.log(this.current);//当前path路径}
}VueRouter.install = function (_Vue) {//1、保存VueVue = _Vue//2、将以下代码延迟到vue实例初始化完毕执行Vue.mixin({beforeCreate() {//判断如果有router属性就执行下方代码if (this.$options.router) {Vue.prototype.$router = this.$options.router;}}})//创建全局组件router-link和router-view//创建router-link组件Vue.component('router-link', {props: {to: {type: String | Object,required: true}},render(h) {// h 有三个参数,第一个参数创建的元素,第二个参数元素具有的属性, 第三个参数元素的内容return h('a', { attrs: { href: '#' + this.to } }, this.$slots.default)}})//创建router-view组件Vue.component('router-view', {// 组件要重新渲染,必须要让数据发生改变的时候,重新去执行render函数render(h) {const router = this.$routerconsole.log(router.routesMap, 'routesMap');let component = router.routesMap[router.current];//组件console.log(component, 'component');//获取到外部传递的路由表return h(component)}})
}export default VueRouter;

接下来实现历史路由模式

我们知道历史路由模式和hash路由模式区别是域名是否存在#因此我们可以通过在路由文件中定义mode属性判断是历史路由模式还是hash路由模式。

先大白话来说一下大概的实现思路(hash和history实现区别):

1、我们通过我们定义的mode判断是hash路由模式还是history路由模式,router-link点击跳转的时候如果是hash路由模式,我们通过render方法返回我们渲染到浏览器域名的数据,hash模式是带#的,因此我们定义href属性需要+#,但是history模式是不带#的,因此我们不需要加#。其次,因为hash模式自动检测我们的域名,因此我们实现历史模式需要手动加一个点击事件,并且需要阻止a链接默认跳转的行为(preventDefault取消默认行为),并且通过pushState方法,实现跳转,这是历史路由模式的方法pushState(obj,title,url),最后设置我们的路由当前path路径为我们跳转的路径。

2、当我们定义好它的router-link全局组件以后,这时我们hash路由模式和history路由模式已经可以基本实现,hash模式带#,history路由模式不带#。但是我们仍要定义一个监听历史路由模式变化的监听事件。这里我们通过mode判断是什么模式从而监听不同的事件,历史路由模式监听需要通过监听popState方法,来判断是否发生变化,它坚挺的是我们浏览器的前进后退的变化。

历史路由模式监听事件为hashchange方法,这个事件是用来监听我们hash值的变化,通过设置两个方法从而设置我们当前路径为对应的path路径。hash模式会通过slice剪切到#,历史路由模式则通过location.pathname获取到当前path路径。

实现如下:

let Vue;class VueRouter {constructor(options) {//保存选项this.options = options;// console.log(options, '路由数据');// 定义一个响应式的变量current,保存当前的hash值let url = location.hash.slice(1,) || '/';// defineReactive定义响应式的对象Vue.util.defineReactive(this, 'current', url)//判断当前的路由模式if (this.options.mode === 'hash') {// hashchange事件来监听hash值的变化, 注意this指向问题addEventListener('hashchange', this.changeHash.bind(this))} else if (this.options.mode === 'history') {// history模式监听的事件叫做:popstate, 监听的是浏览器左上角的两个小箭头的变化addEventListener('popstate', this.changeHistory.bind(this))}this.routesMap = {};//对routes中的对象做一个映射:routesMap = {'/home':component,'/list': component}this.options.routes.forEach(route => {this.routesMap[route.path] = route.component})}//监听hash值变化的函数changeHash() {//通过location.hash来获取到hash值this.current = location.hash.slice(1,)console.log(this.routesMap, '555');// console.log(this.current);//当前path路径}// historychangeHistory() {// console.log(location.pathname)this.current = location.pathname;}}

VueRouter.install = function (_Vue) {//1、保存VueVue = _Vue//2、将以下代码延迟到vue实例初始化完毕执行Vue.mixin({beforeCreate() {//判断如果有router属性就执行下方代码if (this.$options.router) {Vue.prototype.$router = this.$options.router;}}})//创建全局组件router-link和router-view//创建router-link组件Vue.component('router-link', {props: {to: {type: String | Object,required: true}},render(h) {// h 有三个参数,第一个参数创建的元素,第二个参数元素具有的属性, 第三个参数元素的内容const router = this.$router;if (router.options.mode === 'hash') {return h('a', { attrs: { href: '#' + this.to } }, this.$slots.default)} else if (router.options.mode === 'history') {return h('a', {attrs: { href: this.to },on: {'click': ev => {// 1. 阻止a链接的默认跳转行为ev.preventDefault()// 2. 调用pushState方法来实现跳转: pushState(obj, title, url)history.pushState({}, '', this.to)// 3. 设置current的值router.current = this.to;}}}, this.$slots.default)}}})//创建router-view组件Vue.component('router-view', {// 组件要重新渲染,必须要让数据发生改变的时候,重新去执行render函数render(h) {const router = this.$routerconsole.log(router.routesMap, 'routesMap');let component = router.routesMap[router.current];//组件console.log(component, 'component');//获取到外部传递的路由表return h(component)}})
}

这一期我们讲解了,如何实现一个基本的hash路由模式以及history路由模式,但是我们通常使用情况下路由嵌套通过一级路由下定义children属性来定义二级路由,这并没有实现,下一期来实现路由嵌套。

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

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

相关文章

Docker中部署Nginx

1.Nginx部署需求 2.操作教程 3.实际步骤 把配置粘过来。

客户端远程启动服务器脚本文件

目录 软件需求 实现 方法1 方法2 方法3 软件需求 有两台计算机&#xff0c;一台是linux客户端&#xff0c;另一台是linux服务器。要求操作员可以在客户端远程启动服务器上的脚本文件&#xff0c;控制服务器。 实现 方法1 客户端通过ssh登录服务器&#xff0c;然后通过…

Cookie、Session、Token的区别

有人或许还停留在它们只是验证身份信息的机制&#xff0c;但是它们之间的关系你真的弄懂了么&#xff1f; 发展史&#xff1a; Coolie: Netscape Communications 公司引入了 Cookie 概念&#xff0c;作为在客户端存储状态信息的一种方法。初始目的是为了解决 HTTP 的无状态性…

Python爬虫:单线程、多线程、多进程

前言 在使用爬虫爬取数据的时候&#xff0c;当需要爬取的数据量比较大&#xff0c;且急需很快获取到数据的时候&#xff0c;可以考虑将单线程的爬虫写成多线程的爬虫。下面来学习一些它的基础知识和代码编写方法。 一、进程和线程 进程可以理解为是正在运行的程序的实例。进…

python爬虫数据解析xpath、jsonpath,bs4

数据的解析 解析数据的方式大概有三种 xpathJsonPathBeautifulSoup xpath 安装xpath插件 打开谷歌浏览器扩展程序&#xff0c;打开开发者模式&#xff0c;拖入插件&#xff0c;重启浏览器&#xff0c;ctrlshiftx&#xff0c;打开插件页面 安装lxml库 安装在python环境中的Scri…

剑指Offer61.扑克牌中的顺子 C++

1、题目描述 从若干副扑克牌中随机抽 5 张牌&#xff0c;判断是不是一个顺子&#xff0c;即这5张牌是不是连续的。2&#xff5e;10为数字本身&#xff0c;A为1&#xff0c;J为11&#xff0c;Q为12&#xff0c;K为13&#xff0c;而大、小王为 0 &#xff0c;可以看成任意数字。…

并发服务器模型,多线程并发

一、多线程并发完整代码 #include <stdio.h> #include <sys/types.h> #include <sys/socket.h> #include <arpa/inet.h> #include <string.h> #include <unistd.h> #include <sys/wait.h> #include <stdlib.h> #include <…

突然让做性能测试?试试RunnerGo

当前&#xff0c;性能测试已经是一名软件测试工程师必须要了解&#xff0c;甚至熟练使用的一项技能了&#xff0c;在工作时可能每次发版都要跑一遍性能&#xff0c;跑一遍自动化。性能测试入门容易&#xff0c;深入则需要太多的知识量&#xff0c;今天这篇文章给大家带来&#…

Rocky Linux更换为国内源

Rocky Linux提供的可供切换的源列表&#xff1a;Mirrors - Mirror Manager 其中以 COUNTRY 列为 CN 的是国内源。 选择其中一个Rocky Linux 源使用帮助 — USTC Mirror Help 文档 操作前请做好备份 对于 Rocky Linux 8&#xff0c;使用以下命令替换默认的配置 sed -e s|^mirr…

新能源汽车电控系统

新能源汽车电控系统主要分为&#xff1a;三电系统电控系统、高压系统电控系统、低压系统电控系统 三电系统电控系统 包括整车控制器、电池管理系统、驱动电机控制器等。 整车控制器VCU 整车控制器作为电动汽车中央控制单元&#xff0c;是整个控制系统的核心&#xff0c;也是…

zabbix监控mysql数据库、nginx、Tomcat

zabbix监控mysql数据库、nginx、Tomcat 一.zabbix监控mysql数据库 1.环境规划 hostIP部署zabbix-server192.168.198.17zabbix服务器搭建zabbix-mysql192.168.198.15zabbix客户端搭建 2.zabbix-server安装部署&#xff08;192.168.198.17&#xff09; 请参考以下配置&#…

Azure概念介绍

云计算定义 云计算是一种使用网络进行存储和处理数据的计算方式。它通过将数据和应用程序存储在云端服务器上&#xff0c;使用户能够通过互联网访问和使用这些资源&#xff0c;而无需依赖于本地硬件和软件。 发展历史 云计算的概念最早可以追溯到20世纪60年代的时候&#x…

mysql 分库分表浅析

分表是分散数据库压力的好方法。 分表&#xff0c;最直白的意思&#xff0c;就是将一个表结构分为多个表&#xff0c;然后&#xff0c;可以再同一个库里&#xff0c;也可以放到不同的库。 当然&#xff0c;首先要知道什么情况下&#xff0c;才需要分表。个人觉得单表记录条数达…

2023河南萌新联赛第(五)场:郑州轻工业大学C-数位dp

链接&#xff1a;登录—专业IT笔试面试备考平台_牛客网 给定一个正整数 n&#xff0c;你可以对 n 进行任意次&#xff08;包括零次&#xff09;如下操作&#xff1a; 选择 n 上的某一数位&#xff0c;将其删去&#xff0c;剩下的左右部分合并。例如 123&#xff0c;你可以选择…

年至年的选择仿elementui的样式

组件&#xff1a;<!--* Author: liuyu liuyuxizhengtech.com* Date: 2023-02-01 16:57:27* LastEditors: wangping wangpingxizhengtech.com* LastEditTime: 2023-06-30 17:25:14* Description: 时间选择年 - 年 --> <template><div class"yearPicker"…

Smart HTML Elements 16.1 Crack

Smart HTML Elements 是一个现代 Vanilla JS 和 ES6 库以及下一代前端框架。企业级 Web 组件包括辅助功能&#xff08;WAI-ARIA、第 508 节/WCAG 合规性&#xff09;、本地化、从右到左键盘导航和主题。与 Angular、ReactJS、Vue.js、Bootstrap、Meteor 和任何其他框架集成。 智…

九、多态(2)

本章概要 构造器和多态 构造器调用顺序继承和清理构造器内部多态方法的行为 协变返回类型使用继承设计 替代 vs 扩展向下转型与运行时类型信息 构造器和多态 通常&#xff0c;构造器不同于其他类型的方法。在涉及多态时也是如此。尽管构造器不具有多态性&#xff08;事实上…

【JavaScript】new 的原理以及实现

网道 - new 命令的原理 使用new命令时&#xff0c;它后面的函数依次执行下面的步骤。 创建一个空对象&#xff0c;作为将要返回的对象实例。将这个空对象的原型&#xff0c;指向构造函数的prototype属性。将这个空对象赋值给函数内部的this关键字。如果构造函数返回了一个对象…

版本动态 | SolidUI 0.2.0 版本发布

SolidUI 一句话生成任何图形 背景 随着文本生成图像的语言模型兴起&#xff0c;SolidUI想帮人们快速构建可视化工具&#xff0c;可视化内容包括2D,3D,3D场景&#xff0c;从而快速构三维数据演示场景。SolidUI 是一个创新的项目&#xff0c;旨在将自然语言处理&#xff08;NLP&…

[SpringCloud] 组件性能优化技巧

Feign 配置优化hystrix配置 优化ribbon 优化Servlet 容器 优化Zuul配置 优化 文章目录 1.Servlet 容器 优化2.Feign 配置优化3.Zuul配置 优化4.hystrix配置 优化5.ribbon 优化 1.Servlet 容器 优化 默认情况下, Spring Boot 使用 Tomcat 来作为内嵌的 Servlet 容器, 可以将 We…