手写nextTick

作者: jie 分类: JavaScript,Vue 发布时间: 2024-12-10 10:39

前言

最近在面试的时候被面试官问到,如何在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中的idapp的容器上,所以我们只需要监听该idapp的容器就能够实现对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]);

	console.log(l.value);

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

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

	// ... 其他代码 ...

	const myNextTick = (fn) => {
		const app = document.getElementById("app");
		const config = {
			childList: true, // 观察目标子节点的变化,是否有添加或者删除
			attributes: true, // 观察属性变动
			subtree: true // 观察后代节点,默认为 false
		};

		const observer = new MutationObserver((mutations) => {
			// console.log(mutations);
			fn();
		});

		observer.observe(app, config);
	};
</script>

<style scoped></style>

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

监听DOM元素变化的原理:

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

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

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

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

使用了原生JavaScript的PromisesetTimeout。这个实现方式模拟了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源码。

发表回复