当前位置:首页 > 技术文章 > 正文内容

因为不会手写nextTick又被面试官diss了

zonemu2个月前 (08-28)技术文章19

最近在面试的时候被面试官问到,如何在vue中获取到最新的DOM元素,瞬间内心狂喜,自信地对面试官说不就是用nextTick嘛,当DOM元素更新时会执行传入nextTick的回调函数,我们在回调函数中就可以获取到最新的DOM了。

嗯,没错,那你手写一个nextTick吧。

痛苦面具,瞬间小脑萎缩了。



什么是nextTick

在 Vue 的生命周期中,有一段时间是异步的,有时候会遇到数据还未挂载到DOM节点上就试图获取该数据那么此时我们获取到的数据并不是最新的,而 nextTick 就是让我们在这段异步时间结束后执行自己的代码的工具。它确保在DOM更新后执行回调 (起到了等待DOM渲染的作用)。

nextTick的用法

<template>
  <div>
    <button @click="add()">添加</button>
    <ul>
      <li v-for="(i, index) in list" ref="l" key="index">{{ i }}</li>
    </ul>
  </div>
</template>

<script setup>
import { nextTick, ref } from "vue";
const l = ref(null)
const list = ref([1, 2, 3])
console.log(l.value);
</script>

大家猜猜这段代码的运行结果是什么?

请看下图:



为什么最后会输出null呢?

这就是一个很经典的问题,在dom节点还未挂载时我们打印该元素,所以打印的结果为null。

我们可以看vue的官方文档给出的生命周期示意图,很好的解释了这一问题。



在这段代码中,生命周期setup是最先执行的,所以在dom节点还未挂载前就会执行console.log(l.value);打印出null。

那么我们怎么在dom节点挂载后再打印呢? 接下来将要请出本文的主角nextTick了

运行下面这段代码

<template>
  <div>
    <button @click="add()">添加</button>
    <ul>
      <li v-for="(i, index) in list" ref="l" key="index">{{ i }}</li>
    </ul>
  </div>
</template>

<script setup>
import { nextTick, ref } from "vue";
const l = ref(null)
const list = ref([1, 2, 3])
nextTick(() => {
  console.log(l.value.length);
})
</script>



运行这段代码,我们可以看到,nextTick等待DOM渲染完毕之后再执行回调函数,这样就完美的解决了问题。

如何手写nextTick

搞懂了上面的这些内容,手写一个nextTick对于大家来说应该可以秒了,接下来让我们一起来看看怎么手写nextTick吧。

根据上面的代码我们可以知道,nextTick函数在DOM节点发生变化时,会执行传入的回调函数。现在的问题就是我们怎么来监听DOM节点的变化呢?

我们都知道在vue中所有的组件最终都是经过编译然后挂载到index.html中的id为app的容器上,所以我们只需要监听该id为app的容器就能够实现对DOM节点更新的监视。

好了,有了以上知识的铺垫,直接开戳

<template>
  <div>
    <button @click="add()">添加</button>
    <ul>
      <li v-for="(i, index) in list" ref="l" key="index">{{ i }}</li>
    </ul>
  </div>
</template>

<script setup>
import { nextTick, ref } from "vue";
const l = ref(null)
const list = ref([1, 2, 3])
nextTick(() => {
  console.log(l.value.length);
})
</script>

在这段代码中,用了原生js的MutationObserver方法,该方法可以用于观察DOM树的变化,在该代码中用实例化了一个观察者对象并指定观察的容器与触发回调函数的条件。

监听 DOM 元素变化的原理:

  1. MutationObserver 被用来观察 DOM 树的变化。
  2. 当 add 方法被调用时,list 的变化会导致列表项的增加,进而引起 DOM 树的变化。
  3. MutationObserver 会检测到这些变化,并在变化发生后调用回调函数,从而执行传入的 fn 函数。

好了,接下来我们来试一下自己手搓的nextTick函数的效果。

可以看到当点击添加按钮时,可以实时获取到当前列表的长度,完美解决。



写到这里,大家觉得这种方法对吗?显然有问题,用MutationObserver方法来监听DOM树的变化时当往根节点中添加其他的元素时也会触发回调函数,所以我们要用事件循环机制来解决。

使用了原生 JavaScript 的 Promise 和 setTimeout。这个实现方式模拟了 Vue.js 的 nextTick 行为,并且可以在浏览器环境中运行。

实现 nextTick 的基本思路:

  1. 异步执行:确保回调是在当前任务完成之后执行的。
  2. 队列管理:如果多次调用 nextTick,确保它们按照顺序执行,而不是并发执行。
  3. 立即执行:如果可能的话,在微任务队列中立即执行(利用 Promise)。
<template>
  <div>

    <button @click="add()">添加</button>

    <ul>
      <li v-for="(i, index) in list" ref="l" :key="index">{{ i }}</li>
    </ul>
  </div>
</template>

<script setup>

import { nextTick, ref } from "vue";


const list = ref([1, 2, 3])

const l = ref(null);

console.log(l.value);


const add = () => {
  list.value.push(list.value.length + 1);

  myNextTick(() => {
    console.log(l.value.length);
  });
};

function myNextTick(callback) {
  return new Promise(resolve => {
    setTimeout(() => {
      if (callback) {
        callback()
      }
      resolve()
    }, 0)
  })
}
</script>

当尝试更新列表长度时,正常获取到最新的dom结构。



以上仅为简单模拟过程,详细过程请参考Vue源码。

总结

本篇文章就到此结束了,希望大家以后再碰到手写nextTick的时候能顺利秒杀

相关文章

7种超轻量级的Linux发行版,能够帮助你找到适合自己的操作系统

Linux是一种非常受欢迎的开源操作系统,而且有许多版本可以选择。有时候,你需要一种超轻量级的Linux发行版,它可以在资源有限的设备上运行,并且能够快速启动。本文将介绍7种超轻量级的Linux发行版...

100行Html5+CSS3+JS代码实现元旦倒计时界面

一、前言2022年到了,祝大家虎年大吉喜气临,昂首摆尾迎春来。双眼圆睁看世界,万水千山尽开颜。胡须翘翘美姿态,人人开心祝平安。巨大身躯摇摆摆,坎坷困境当笑谈。愿你虎年万事顺,吉星高照旺旺旺!二、202...

为了成为Claude Code高手,我雇了个AI当教练

在AI编程的浪潮中,如何高效提升编程能力成为许多开发者关注的焦点。本文作者通过自身实践,分享了如何利用AI工具(如Claude Code和Cursor)进行编程协作,并通过“学习导航器”提示词实现高效...

js中数组filter方法的使用和实现(js实现数组的indexof方法)

定义filter() 方法创建一个新数组, 其包含通过所提供函数实现的测试的所有元素。语法var newArray = arr.filter(callback(element[, index[, se...

面试官-如何实现数组和 List 之间的转换?

数组和List是Java开发中常见的两种数据结构,那么如何实现二者之间的快速转换就成了面试官常问的考点之一,下面我们我们就来从数组转List和List转数组两个方面来展开介绍一下。数组转List方法...

WordPress 内置的数组处理相关函数大全

我们使用 WordPress 开发的时候,有很大一部分的工作和数组处理有关,WordPress 本身也内置了一些非常方便的数组处理函数,今天给大家罗列一下,也方便自己以后写代码的时候查询。wp_par...