前言

在上一篇 [手撕Vue-Router-实现router-link] 中,我们实现了 router-link 组件,这一篇我们来实现 router-view 组件。

实现思路

router-view 其实与 router-link 是同一个世界,同一个梦想的,那么我就废话不多说了,直接上代码。

实现代码

回到我们的 NueRouter.install 方法中,直接编写 Vue.component('router-view', {}),这里我们直接使用了 Vue.component 方法,这个方法是 Vue 内部提供的,用于注册全局组件的方法。

Vue.component('router-view', {});

注册好了全局组件之后我们就可以使用这个组件了,我们在 App.vue 中使用这个组件。

接下来我们要做的事情就是根据 router-link 中的 to 属性,来获取对应的组件,然后渲染到 router-view 中。

那么怎么渲染呢?我们可以使用 render 函数来渲染,render 函数是 Vue 内部提供的,用于渲染组件的函数。render 函数接收一个 h 函数作为参数,h 函数用于渲染组件。

代码如下:

Vue.component('router-view', {
render(h) {
return h();
}
});

那么我们就可以在 h 函数中,根据路由地址来获取对应的组件,然后渲染到 router-view 中。

那么问题就来了我们拿到对应的组件呢?首先我们需要获取到路由信息,然后在拿到当前的路由地址,然后拿到当前的路由地址,去我提取的路由信息中,找到对应的组件。

好了,我们知道了我们要做什么,那么我们就开始做吧。

首先我们需要获取到路由信息,我们可以通过 const routesMap = this._self.$router.routesMap; 来获取到路由信息。

然后我们需要获取到当前的路由地址,我们可以通过 const currentPath = this._self.$route.currentPath 来获取到当前的路由地址。

然后我们就可以通过 routesMap[currentPath] 来获取到对应的组件了。

那么我们就可以在 h 函数中,根据路由地址来获取对应的组件,然后渲染到 router-view 中。

代码如下:

Vue.component('router-view', {
render(h) {
const routesMap = this._self.$router.routesMap;
const currentPath = this._self.$route.currentPath;
const currentComponent = routesMap[currentPath];
return h(currentComponent);
}
});

好了,我们的 router-view 组件就实现了,我们来看一下效果。浏览器自行测试。

我在测试过程中,我点击了 首页关于 两个按钮,发现并没有渲染出对应的组件,这是为什么呢?并不是我们的代码有问题,而是在渲染网页的时候,首先它会先渲染组件,然后再执行 load 事件,所以就导致了我们在渲染组件的时候,我们还没有保存 currentPath,因为 currentPath 是在 load 事件中保存的,所以我们在渲染组件的时候,是获取不到 currentPath 的,怎么验证这个问题呢?我们可以在 load 打印一个内容,然后在 render 中,也打印一下 内容,并且将 currentPath 也打印出来,我们就可以看到这个问题了。

打开浏览器,点击首页按钮,控制台打印如下:

可以看到,我们在渲染组件的时候,currentPath 是 null,这就是我们的问题所在。

解决方案

那么我们怎么解决这个问题呢?其实很简单,我们只需要监听 currentPath 的变化,然后在 currentPath 变化的时候,重新渲染组件就可以了。

那么怎么让 currentPath 变化之后,重新渲染组件呢?默认情况下 Vue 是双向数据绑定的,默认情况下只要数据发生变化,就会重新渲染组件。

但是有一个前提条件,就是数据必须是响应式的(也就是说 currentPath 是一个双向绑定的数据),那么怎么让 currentPath 变成一个响应式的数据呢?其实可以在我们往 Vue 实例中添加 $router or $route 的时候,将 currentPath 变成一个响应式的数据。

那么怎么将 currentPath 变成一个响应式的数据呢?我们可以使用 Vue 内部提供的 Vue.util.defineReactive 方法,这个方法可以将一个普通的数据变成一个响应式的数据。

那么我们就可以在我们往 Vue 实例中添加 $router or $route 的时候,将 currentPath 变成一个响应式的数据。

代码如下:

Vue.util.defineReactive(this, 'xxx', this.$router);

好了,我们的 currentPath 变成了一个响应式的数据,那么我们就可以在 currentPath 变化的时候,重新渲染组件了。

我们的 router-view 组件就实现了。浏览器自行测试。

总结

主要就是使用 Vue.component 方法来注册全局组件,然后使用 render 函数来渲染组件,然后在 h 函数中,根据路由地址来获取对应的组件,然后渲染到 router-view 中。

还有就是在渲染组件的时候,currentPath 是 null,因为而是在渲染网页的时候,首先它会先渲染组件,然后再执行 load 事件,所以就导致了我们在渲染组件的时根据 currentPath 获取不到对应的组件。

解决方案就是将 currentPath 变成一个响应式的数据,默认情况下 Vue 是双向数据绑定的,默认情况下只要数据发生变化,就会重新渲染组件, 这样一来就达到了我们的目的。

最后

大家好我是 BNTang, 一个热爱分享的技术的开发者,如果大家觉得我的文章对你有帮助的话,可以关注我的公众号 JavaBoyL,我会在公众号中分享一些IT技术和一些个人的见解,谢谢大家的支持。

手撕Vue-Router-实现router-view的更多相关文章

  1. Vue 组件之 Router

    Vue 组件之 Router Vue 开发单页应用的时候,免不了使用Vue组件.在单页应用上如何进行组件切换? 结构如下图所示: 主页面包含Foo组件与Bar组件,在主页面中可以进行Foo与 Bar的 ...

  2. vue路由请求 router

    创建一个Router.js文件 // 路由请求//声明一个常量设置路菜单// import Vue from "vue/types/index";import Vue from ' ...

  3. vue学习之router

    路由文档:https://router.vuejs.org/zh/guide/ 使用vue做spa应用的话,一定会涉及到路由. 安装 安装router插件 npm install vue-router ...

  4. vue项目中router路由配置

    介绍 路由:控制组件之间的跳转,不会实现请求.不用页面刷新,直接跳转-切换组件>>> 安装 本地环境安装路由插件vue-router:    cnpm install vue-rou ...

  5. vue中使用router全局守卫实现页面拦截

    一.背景 在vue项目中使用vue-router做页面跳转时,路由的方式有两种,一种是静态路由,另一种是动态路由.而要实现对路由的控制需要使用vuex和router全局守卫进行判断拦截(安全问题文章最 ...

  6. Vue的路由Router之导航钩子和元数据及匹配

    一.文件结构 二.vue.js 打开此链接 https://cdn.bootcss.com/vue/2.6.10/vue.js 复制粘贴页面的所有内容 三.vue-router.js 打开此链接  h ...

  7. vue全家桶router、vuex、axios

    main.js import Vue from 'vue' import App from './App' import router from './router' import store fro ...

  8. vue & vue router & dynamic router

    vue & vue router & dynamic router https://router.vuejs.org/guide/essentials/dynamic-matching ...

  9. 【面试题】Vue中的$router 和 $route的区别

    Vue中的$router 和 $route的区别 点击视频讲解更加详细 this.$route:当前激活的路由的信息对象.每个对象都是局部的,可以获取当前路由的 path, name, params, ...

  10. 手写vue路由

    目录 一.简易demo 二.Vue-Router传参方式 三.进阶-路由导航 一.简易demo // routes注册 import Vue from "vue"; // impo ...

随机推荐

  1. Java异步编程详解

    在现代应用程序开发中,异步编程变得越来越重要,特别是在处理I/O密集型任务时.Java提供了一套强大的异步编程工具,使得开发者能够更有效地处理并发任务.本篇博文将深入探讨Java中异步编程的方方面面, ...

  2. JSP使用MySQL数据库报错java.lang.ClassNotFoundException: com.mysql.jdbc.Driver

    错误 在JavaWeb中的JSP中使用MySQL数据库报错找不到 com.mysql.jdbc.Driver . 错误日志如下: java.lang.ClassNotFoundException: c ...

  3. 洛谷P1174 打砖块 | CCPC2021网络赛8.28 1011 动态规划 分组背包

    本文学习自洛谷社区 喜提CCPC2021网络赛原题 题意相当于是要在每一列中选若干个砖块打掉,消耗所需的子弹数并得到对应的得分.最大化k个子弹能得到的最大得分. 预处理出第 \(i\) 列 \(j\) ...

  4. Linux复制安装 jdk 环境

    转载请注明出处: 最近在弄服务器环境,发现可以通过复制已安装 jdk 的服务器配置到新的服务器,并配置服务器环境变量配置文件就可以完成. 操作步骤如下: 1. 查看以安装jdk服务器的环境配置,并复制 ...

  5. Python追踪内存占用

    技术背景 当我们需要对python代码所占用的内存进行管理时,首先就需要有一个工具可以对当前的内存占用情况进行一个追踪.虽然在Top界面或者一些异步的工具中也能够看到实时的内存变化,还有一些工具可以统 ...

  6. WPF|如何在 WPF 中设计漂亮的社交媒体信息仪表板

    1. 效果展示 先来直接欣赏效果: 2. 准备 创建一个WPF工程,比如站长使用 .NET 7 创建名为 Dashboard3 的WPF项目,添加一些图片资源,项目目录如下: 2.1 图片资源 可在网 ...

  7. [转帖]clickhouse 超底层原理& 高可用集群 实操(史上最全)

    https://www.cnblogs.com/crazymakercircle/p/16718469.html 文章很长,而且持续更新,建议收藏起来,慢慢读!疯狂创客圈总目录 博客园版 为您奉上珍贵 ...

  8. [转帖]global cache cr request等待事件分析及优化

    在RAC环境中,和全局调整缓存相关的最常见的等待事件无非就是:global cache cr request,global cache busy和equeue 在XX电信做了一次数据库巡检中发现,sp ...

  9. [转帖]tidb 搭建私有镜像库

    https://docs.pingcap.com/zh/tidb/stable/tiup-mirror 在构建私有云时,通常会使用隔离的网络环境,此时无法访问 TiUP 的官方镜像.因此,TiUP 提 ...

  10. [转帖] JVM诊断命令jcmd介绍

    https://www.cnblogs.com/codelogs/p/16535451.html 简介# 从JDK7开始,jdk提供了一个方便扩展的诊断命令jcmd,用来取代之前比较分散的jdk基础命 ...