Vue强制组件重新渲染的方法讨论


Posted in Javascript onFebruary 03, 2020

有时候,依赖 Vue 响应方式来更新数据是不够的,相反,我们需要手动重新渲染组件来更新数据。或者,我们可能只想抛开当前的DOM,重新开始。那么,如何让Vue以正确的方式重新呈现组件呢?

强制 Vue 重新渲染组件的最佳方法是在组件上设置:key。 当我们需要重新渲染组件时,只需更 key 的值,Vue 就会重新渲染组件。

这是一个非常简单的解决方案。

当然,你可能会对其他方式会更感兴趣:

简单粗暴的方式:重新加载整个页面
不妥的方式:使用 v-if
较好的方法:使用Vue的内置forceUpdate方法
最好的方法:在组件上进行 key 更改

简单粗暴的方式:重新加载整个页面

这相当于每次你想关闭应用程序时都要重新启动你的电脑。

这种方式或许有用,但这是一个非常糟糕的解决方案,不要这样做,我们来看看更好的方法。

不妥的方式:使用 v-if

v-if指令,该指令仅在组件为true时才渲染。 如果为false,则该组件在DOM中不存在。

来看看,v-if 是怎么工作的,在template中,添加v-if指令:

<template>
 <my-component v-if="renderComponent" />
</template>

在script 中,使用nextTick的方法

<script>
 export default {
  data() {
   return {
    renderComponent: true,
   };
  },
  methods: {
   forceRerender() {
    // 从 DOM 中删除 my-component 组件
    this.renderComponent = false;
    
    this.$nextTick(() => {
     // 在 DOM 中添加 my-component 组件
     this.renderComponent = true;
    });
   }
  }
 };
</script>

上面的过程大致如下:

刚开始 renderComponent设置为true,因此渲染 my-component 组件
当我们调用forceRerender时,我们立即将renderComponent设置为false
我们停止渲染my-component,因为v-if指令现在计算结果为false
在nextTick方法中将renderComponent设置回true
当v-if指令的计算结果为true时,再次渲染my-component

在这个过程中,有两个部分比较重要

首先,我们必须等到nextTick,否则我们不会看到任何变化。

在Vue中,一个 tick 是一个DOM更新周期。Vue将收集在同一 tick 中进行的所有更新,在 tick 结束时,它将根据这些更新来渲染 DOM 中的内容。如果我们不等到next tick,我们对renderComponent的更新就会自动取消,什么也不会改变。

其次,当我们第二次渲染时,Vue将创建一个全新的组件。 Vue 将销毁第一个,并创建一个新的,这意味着我们的新my-component将像正常情况一样经历其所有生命周期-created,mounted等。

另外,nextTick 可以与 promise 一起使用:

forceRerender() {
 // 从 DOM 中删除 my-component 组件
 this.renderComponent = false;

 this.$nextTick().then(() => {
  this.renderComponent = true;
 });
}

不过,这并不是一个很好的解决方案,所以,让我们做 Vue 想让我们做的

较好的方法:forceUpdate 方法

这是解决这个问题的两种最佳方法之一,这两种方法都得到了Vue的官方支持。

通常情况下,Vue 会通过更新视图来响应依赖项中的更改。然而,当我们调用forceUpdate时,也可以强制执行更新,即使所有依赖项实际上都没有改变。

下面是大多数人使用这种方法时所犯的最大错误。

如果 Vue 在事情发生变化时自动更新,为什么我们需要强制更新呢?

原因是有时候 Vue 的响应系统会让人感到困惑,我们认为Vue会对某个属性或变量的变化做出响应,但实际上并不是这样。在某些情况下,Vue的响应系统根本检测不到任何变化。

所以就像上一个方法,如果你需要这个来重新渲染你的组件,可能有一个更好的方法。

有两种不同的方法可以在组件实例本身和全局调用forceUpdate:

// 全局
import Vue from 'vue';
Vue.forceUpdate();

// 使用组件实例
export default {
 methods: {
  methodThatForcesUpdate() {
   // ...
   this.$forceUpdate();
   // ...
  }
 }
}

重要提示:这不会更新任何计算属性,调用forceUpdate仅仅强制重新渲染视图。

最好的方法:在组件上进行 key 更改

在许多情况下,我们需要重新渲染组件。

要正确地做到这一点,我们将提供一个key属性,以便 Vue 知道特定的组件与特定的数据片段相关联。如果key保持不变,则不会更改组件,但是如果key发生更改,Vue 就会知道应该删除旧组件并创建新组件。

正是我们需要的!

但是首先,我们需要绕一小段路来理解为什么在Vue中使用key。

为什么我们需要在 Vue 中使用 key
一旦你理解了这一点,那么这是了解如何以正确方式强制重新渲染的很小的一步。

假设我们要渲染具有以下一项或多项内容的组件列表:

有本地的状态
某种初始化过程,通常在created或mounted钩子中
通过jQuery或普通api进行无响应的DOM操作
如果你对该列表进行排序或以任何其他方式对其进行更新,则需要重新渲染列表的某些部分。 但是,不会希望重新渲染列表中的所有内容,而只是重新渲染已更改的内容。

为了帮助 Vue 跟踪已更改和未更改的内容,我们提供了一个key属性。 在这里使用数组的索引,因为索引没有绑定到列表中的特定对象。

const people = [
 { name: 'Evan', age: 34 },
 { name: 'Sarah', age: 98 },
 { name: 'James', age: 45 },
];

如果我们使用索引将其渲染出来,则会得到以下结果:

<ul>
 <li v-for="(person, index) in people" :key="index">
  {{ person.name }} - {{ index }}
 </li>
</ul>

// Outputs
Evan - 0
Sarah - 1
James - 2

如果删除Sarah,得到:

Evan - 0
James - 1

与James关联的索引被更改,即使James仍然是James。 James会被重新渲染,这并不是我们希望的。

所以这里,我们可以使用唯一的 id 来作为 key

const people = [
 { id: 'this-is-an-id', name: 'Evan', age: 34 },
 { id: 'unique-id', name: 'Sarah', age: 98 },
 { id: 'another-unique-id', name: 'James', age: 45 },
];

<ul>
 <li v-for="person in people" :key="person.id">
  {{ person.name }} - {{ person.id }}
 </li>
</ul>

在我们从列表中删除Sarah之前,Vue删除了Sarah和James的组件,然后为James创建了一个新组件。现在,Vue知道它可以为Evan和James保留这两个组件,它所要做的就是删除Sarah的。

如果我们向列表中添加一个person,Vue 还知道可以保留所有现有的组件,并且只需要创建一个新组件并将其插入正确的位置。这是非常有用的,当我们有更复杂的组件,它们有自己的状态,有初始化逻辑,或者做任何类型的DOM操作时,这对我们很有帮助。

所以接下来看看,如果使用最好的方法来重新渲染组件。

更改 key 以强制重新渲染组件
最后,这是强制Vue重新渲染组件的最佳方法(我认为)。

我们可以采用这种将key分配给子组件的策略,但是每次想重新渲染组件时,只需更新该key即可。

这是一个非常基本的方法

<template>
 <component-to-re-render :key="componentKey" />
</template>


export default {
 data() {
  return {
   componentKey: 0,
  };
 },
 methods: {
  forceRerender() {
   this.componentKey += 1; 
  }
 }
}

每次forceRerender被调用时,我们的componentKey都会改变。当这种情况发生时,Vue将知道它必须销毁组件并创建一个新组件。我们得到的是一个子组件,它将重新初始化自身并“重置”其状态。

如果确实需要重新渲染某些内容,请选择key更改方法而不是其他方法。

Javascript 相关文章推荐
javascript下阻止表单重复提交、防刷新、防后退
Aug 17 Javascript
js 动态选中下拉框
Nov 26 Javascript
使用jQuery操作Cookies的实现代码
Oct 09 Javascript
js返回上一页并刷新代码整理
Dec 21 Javascript
JS批量修改PS中图层名称的方法
Jan 26 Javascript
Node.js Streams文件读写操作详解
Jul 04 Javascript
javascript事件的绑定基础实例讲解(34)
Feb 14 Javascript
微信小程序学习之数据处理详解
Jul 05 Javascript
Vue中Quill富文本编辑器的使用教程
Sep 21 Javascript
详解微信UnionID作用
May 15 Javascript
教你30秒发布一个TypeScript包到NPM的方法步骤
Jul 22 Javascript
TypeScript的安装、使用、自动编译的实现
Apr 10 Javascript
JavaScript中的类型检查
Feb 03 #Javascript
Vue的Eslint配置文件eslintrc.js说明与规则介绍
Feb 03 #Javascript
压缩Vue.js打包后的体积方法总结(Vue.js打包后体积过大问题)
Feb 03 #Javascript
24个解决实际问题的ES6代码片段(小结)
Feb 02 #Javascript
浅谈vuex为什么不建议在action中修改state
Feb 02 #Javascript
vuex+axios+element-ui实现页面请求loading操作示例
Feb 02 #Javascript
vue实现的封装全局filter并统一管理操作示例
Feb 02 #Javascript
You might like
php实现ping
2006/10/09 PHP
php实现邮件发送并带有附件
2014/01/24 PHP
php过滤输入操作之htmlentities与htmlspecialchars用法分析
2017/02/17 PHP
yepnope.js 异步加载资源文件
2011/09/08 Javascript
ie中js创建checkbox默认选中问题探讨
2013/10/21 Javascript
js判断元素是否隐藏的方法
2014/06/09 Javascript
JavaScript根据CSS的Media Queries来判断浏览设备的方法
2016/05/10 Javascript
微信小程序网络请求的封装与填坑之路
2017/04/01 Javascript
js实现上传并压缩图片效果
2018/01/10 Javascript
使用百度地图实现地图网格的示例
2018/02/06 Javascript
修改Nodejs内置的npm默认配置路径方法
2018/05/13 NodeJs
js最实用string(字符串)类型的使用及截取与拼接详解
2019/04/26 Javascript
vue安装遇到的5个报错及解决方法
2019/06/12 Javascript
javascript中的闭包概念与用法实践分析
2019/07/26 Javascript
用实例说明python的*args和**kwargs用法
2013/11/01 Python
python实现的守护进程(Daemon)用法实例
2015/06/02 Python
python实现备份目录的方法
2015/08/03 Python
python文件的md5加密方法
2016/04/06 Python
简单易懂的python环境安装教程
2017/07/13 Python
详解如何使用Python编写vim插件
2017/11/28 Python
对DataFrame数据中的重复行,利用groupby累加合并的方法详解
2019/01/30 Python
Django中提供的6种缓存方式详解
2019/08/05 Python
python sqlite的Row对象操作示例
2019/09/11 Python
Python Opencv提取图片中某种颜色组成的图形的方法
2019/09/19 Python
国际领先的学术出版商:Springer
2017/01/11 全球购物
欧洲品牌瓷器餐具网上商店:Porzellantreff.de
2018/04/04 全球购物
微软中国官方商城:Microsoft Store中国
2018/10/12 全球购物
Hoover胡佛官网:美国吸尘器和洗地机品牌
2019/01/09 全球购物
某个公司的Java笔面试题
2016/03/11 面试题
外贸销售员求职的自我评价
2013/11/23 职场文书
毕业生个人求职自荐信
2014/02/26 职场文书
文艺晚会策划方案
2014/06/11 职场文书
干货:如何写好工作总结报告!
2019/05/10 职场文书
如何理解Vue前后端数据交互与显示
2021/05/10 Vue.js
写好Python代码的几条重要技巧
2021/05/21 Python
简述Java中throw-throws异常抛出
2021/08/07 Java/Android