nextTick()

在jQuery中,如果我们要生成一个ul-li的列表元素,我们也不会在循环体中每生成一个li就将它插入到ul中,而是在循环体内拼接每个li,待循环体结束后,再一并添加到ul元素上进行视图渲染。这样是为了避免页面不断的进行重排和重绘,优化页面性能。

var list = [1,2,3,4,5,6]

// bad
for (var i = 0; i < list.length; i++) {
$("<li></li>").text(list[i]).appendTo("#one")
} // good
var temp = ""
for (var i = 0; i < list.length; i++) {
temp += `<li>${list[i]}</li>`
}
$("#two").append(temp)

同样,在Vue中,不可能监控到一个数据更新就马上去刷新视图,而是异步执行 DOM 更新。

当观察到数据变化,Vue 会开启一个队列,用来缓冲在同一事件循环中发生的所有数据改变。如果同一个 watcher 被多次触发,只会被推入到队列中一次。然后,在下一个的事件循环前的“tick”中,Vue 再使用刷新后队列中的数据去执行渲染工作。这种使用队列并在缓冲时去除重复数据对于避免不必要的计算和 DOM 操作是非常重要的,很大的提升了页面性能。

所以当我们在this.someData = newValue时,并没有马上在页面中渲染newValue。所以如果此时我们从DOM元素中获取相应的值是没有改变的。

<div id="tick">
<p ref="msg">{{ msg }}</p>
<button @click="updateMsg">更新msg</button>
</div>
var vm = new Vue({
el: "#tick",
data: {
msg: 123
},
methods: {
updateMsg() {
this.msg = 456;
console.log("this.msg",this.msg) // 456 数值是已更新
console.log("msg.textContent", this.$refs.msg.textContent) // 123 视图未更新
}
}
})

如果实际需求确实需要操作数据变化后相关的DOM,VUE提供了nextTick( callback )方法来执行DOM渲染完成后的回调。可以使用全局的Vue.nextTick( callback ),也可以使用实例对象的提供的this.$nextTick( callback )

上面的例子改写下:

var vm = new Vue({
el: "#tick",
data: {
msg: 123
},
methods: {
updateMsg() {
this.msg = 456;
console.log("this.msg",this.msg) // 456 数值是已更新
console.log("msg.textContent", this.$refs.msg.textContent) // 123 视图未更新
this.$nextTick(() => {
console.log("nextTick", this.$refs.msg.textContent) // 456 此时视图已更新
})
}
}
})

现在,回头再仔细回味下解释vue文档中的异步更新队列中这段解释:

只要观察到数据变化,Vue 将开启一个队列,并缓冲在同一事件循环中发生的所有数据改变。如果同一个 watcher 被多次触发,只会被推入到队列中一次。

同一个数值在同一事件中被多次改变,队列中只会保留最后值用来执行视图更新,并且执行页面渲染时机是当前事件执行结束后。

我们验证下:


var vm = new Vue({
el: "#tick",
data: {
msg: 123
},
methods: {
updateMsg() {
this.msg = 456;
console.log("first_this.msg",this.msg)
console.log("first_this.$refs.msg.textContent", this.$refs.msg.textContent)
this.$nextTick(() => {
console.log("first_nextTick", this.$refs.msg.textContent)
}) for (let i=0; i < 5000; i++) {
console.log('i')
} this.msg = 789;
console.log("second_this.msg",this.msg)
console.log("second_this.$refs.msg.textContent", this.$refs.msg.textContent)
this.$nextTick(() => {
console.log("second_nextTick", this.$refs.msg.textContent)
})
}
}
})
first_this.msg 456
first_this.$refs.msg.textContent 123
5000 i
second_this.msg 789
second_this.$refs.msg.textContent 123
first_nextTick 789
second_nextTick 789

观察页面,并没有出现456后再变成789,而是直接变成789,两次的赋值只取了最后一次数值用来渲染,并且两次nextTick回调打印的都是同一个值。所以页面的渲染是在整个updateMsg事件执行后进行的。

vue-learning:11 -js-nextTick()的更多相关文章

  1. 从壹开始前后端分离 [ Vue2.0+.NET Core2.1] 十五 ║Vue基础:JS面向对象&字面量& this字

    缘起 书接上文<从壹开始前后端分离 [ Vue2.0+.NET Core2.1] 十四 ║ VUE 计划书 & 我的前后端开发简史>,昨天咱们说到了以我的经历说明的web开发经历的 ...

  2. vue初学:基础概念

    一.vue使用步骤: 1.引包vue.js 2.html中写要操作的DOM节点 3.创建vue对象:new Vue({options}); 4.配置options:el:(要操作的对象,用选择器,同j ...

  3. Vue源码解析之nextTick

    Vue源码解析之nextTick 前言 nextTick是Vue的一个核心功能,在Vue内部实现中也经常用到nextTick.但是,很多新手不理解nextTick的原理,甚至不清楚nextTick的作 ...

  4. vue中config/index.js:配置的详细理解

    当我们需要和后台分离部署的时候,必须配置config/index.js: 用vue-cli 自动构建的目录里面  (环境变量及其基本变量的配置) var path = require('path') ...

  5. vue-learning:22 - js - directives

    directives 在讲解视图层指令时,我们讲到ref特性,使用它我们可以获取当前DOM元素对象,以便执行相关操作. <div id="app"> <input ...

  6. vue开发:前端项目模板

    简介 vue-cli创建vue项目,整合vuex.vue-router.axios.element-ui 项目模板下载地址 创建项目 使用vue-cli创建项目,功能选择:Babel.Router.v ...

  7. iosselect:一个js picker项目,在H5中实现IOS的select下拉框效果

    具体文档和demo可以访问github:https://github.com/zhoushengmufc/iosselect 移动端浏览器对于select的展示样式是不一致的,ios下是类似原生的pi ...

  8. iosselect:一个js picker项目,在H5中实现IOS的下拉效果

    iosselect是在webapp下的一个picker组件,可以轻松实现各类选择器效果.比如地区选择 时间选择 日期选择等. 下面是一个地址选择器demo截图,可以访问:http://zhoushen ...

  9. Javascript模块化编程(三):require.js的用法

    Javascript模块化编程(三):require.js的用法 原文地址:http://www.ruanyifeng.com/blog/2012/11/require_js.html 作者: 阮一峰 ...

  10. JS组件系列——开源免费图表组件:Chart.js

    前言:最近被开源免费得有点上火了,各种组件首先想到的就是是开源否.是否免费.是否和bootstrap风格一致.想着以后做报表肯定要用到图表组件的,于是在Bootstrap中文网上面找到了Chart.j ...

随机推荐

  1. app被Rejected 的各种原因翻译。这个绝对有用

    1. Terms and conditions(法律与条款) 1.1  As a developer of applications for the App Store you are bound b ...

  2. AT3728 Squirrel Migration

    AT3728 Squirrel Migration 就是给每个点分配两个匹配点(自环除外) 考虑最大值 考虑极限情况:每个边的贡献是min(sz[u],sz[v])*2 证明存在方案: 发现,如果哪边 ...

  3. 20.libgdx,stage中默认相机的使用

    主要思路: 通过查资料得知,stage中的默认封装的相机为OrthographicCamera,要操纵该相机,直接把他转化为OrthographicCamera即可使用 但是这会导致一个问题,即原本固 ...

  4. 用div漂浮快实现与表单无关的多文件上传功能。

    我项目有这个需求,多文件上传,而且要及时显示到表单上,这样的话就不能与表单相关. 由于我对前端不熟,我就实现了这么一个功能,通过button触发一个div漂浮块,然后多文件上传,之后通过js把文件名显 ...

  5. 2018-9-30-C#-从零开始写-SharpDx-应用-画三角

    title author date CreateTime categories C# 从零开始写 SharpDx 应用 画三角 lindexi 2018-09-30 18:30:14 +0800 20 ...

  6. HZOJ Weed

    作者的题解: 如果一段操作被执行,会对整个栈有什么影响呢? 把栈弹出若干个数后再插入若干个数. 线段树: 每个点纪录三个值:执行完这段操作后会删多少个,再插多少个,插的和一共是多少. 合并值时再用一个 ...

  7. 21Hash算法以及暴雪Hash

    一:哈希表简介 哈希表是一种查找效率极高的数据结构,理想情况下哈希表插入和查找操作的时间复杂度均为O(1),任何一个数据项可以在一个与哈希表长度无关的时间内计算出一个哈希值(key),然后在常量时间内 ...

  8. 亿级消息系统的核心存储:Tablestore发布Timeline 2.0模型

    背景 互联网快速发展的今天,社交类应用.消息类功能大行其道,占据了大量网络流量.大至钉钉.微信.微博.知乎,小至各类App的推送通知,消息类功能几乎成为所有应用的标配.根据场景特点,我们可以将消息类场 ...

  9. 在VirtualBox下安装linux操作系统

    目标:在linux服务器上部署Java开发的网站 工具 VirtualBox-4.3.8:下载后安装. linux系统镜像: Centos国内镜像文件下载地址: http://centos.ustc. ...

  10. java+内存分配及变量存储位置的区别

    Java内存分配与管理是Java的核心技术之一,之前我们曾介绍过Java的内存管理与内存泄露以及Java垃圾回收方面的知识,今天我们再次深入Java核心,详细介绍一下Java在内存分配方面的知识.一般 ...