3 分钟带你了解 Vue3 的 nextTick()

前言

Vue 实现响应式并不是数据发生变化之后 DOM 立即变化,而是按一定的策略进行 DOM 的更新。简单来说,Vue在修改数据后,视图不会立刻更新,而是等同一事件循环中的所有数据变化完成之后,再统一进行视图更新,nextTick()方法就会在这样的DOM更新循环结束后调用指定的函数。

概念

在 Vue 3 中nextTick()方法的行为和 Vue 2 中基本相似,但它进行了一些优化,以提高性能和稳定性。nextTick()通过利用微任务(microtask)来确保在 DOM 更新之后执行回调,从而避免在同一事件循环中执行 DOM 操作。简单的理解是,当数据更新了,在dom中渲染后,自动执行函数。

方法的使用

nextTick()的基本使用

javascript 复制代码
// 使用 nextTick() 方法
import { ref, nextTick } from 'vue';
 
const myValue = ref('initial value');
 
// 在 DOM 更新后执行回调
nextTick(() => {
  console.log('DOM 已更新');
});

在onMounted中使用

在该示例中,nextTick()写在onMounted钩子函数中能确保回调在组件首次挂载后执行,常用于处理与初始渲染相关的逻辑。

javascript 复制代码
// 示例 2: 在 setup() 中使用 nextTick()
import { ref, onMounted, nextTick } from 'vue';
 
export default {
  setup() {
    const myValue = ref('initial value');
 
    onMounted(() => {
      // 在 DOM 更新后执行回调
      nextTick(() => {
        console.log('DOM 已更新');
      });
    });
 
    return {
      myValue,
    };
  },
};

nextTick() 的异步执行

nextTick()中的回调函数是异步执行的。这意味着即使在同一事件循环中调用多次nextTick(),它们的回调函数也会按照调用顺序异步执行。

javascript 复制代码
// 示例 3: nextTick() 的异步执行
import { ref, nextTick } from 'vue';
 
const myValue = ref('initial value');
 
// 在 DOM 更新后执行第一个回调
nextTick(() => {
  console.log('第一个回调执行');
});
 
// 在 DOM 更新后执行第二个回调
nextTick(() => {
  console.log('第二个回调执行');
});

nextTick() 的使用场景

在数据变化后等待DOM更新

这是nextTick()最常见的用途。例如,你可能更改了一个数据属性,该属性控制一个元素的可见性。然后你可能想要等待DOM更新以便可以获取该元素的新的宽度或高度。在这种情况下,你可以使用nextTick()来确保你的代码在DOM更新后执行。

javascript 复制代码
<template>  
  <div ref="myDiv">{{ message }}</div>  
</template>  
  
<script>  
import { ref, nextTick } from 'vue';  
  
export default {  
  setup() {  
    const message = ref('Hello Vue!');  
    const myDiv = ref(null);  
  
    function updateMessage() {  
      message.value = 'Updated!';  
      nextTick(() => {  
        console.log(myDiv.value.textContent); // 输出: Updated!  
      });  
    }  
  
    return {  
      message,  
      myDiv,  
      updateMessage  
    };  
  }  
};  
</script>

在创建或销毁组件后等待DOM更新

当创建或销毁Vue组件时,Vue需要时间来更新DOM。如果你需要立即访问新创建或已销毁的DOM元素,你可能会遇到问题,因为DOM可能还没有更新。在这种情况下,你可以使用nextTick()来确保你的代码在DOM更新后执行。

javascript 复制代码
<template>  
  <div>  
    <button @click="toggleComponent">Toggle Component</button>  
    <ChildComponent v-if="showChild" ref="childComponentRef" />  
  </div>  
</template>  
  
<script>  
import { ref, nextTick } from 'vue';  
import ChildComponent from './ChildComponent.vue';  
  
export default {  
  components: {  
    ChildComponent  
  },  
  setup() {  
    const showChild = ref(false);  
    const childComponentRef = ref(null);  
// 在这个例子中,我们有一个ChildComponent,它根据showChild的值来条件性地渲染。
//当我们点击按钮时,toggleComponent方法会被调用,它切换showChild的值。
//然后,我们使用nextTick()来等待Vue的DOM更新队列清空,确保如果ChildComponent被创建,
//它现在已经被挂载到DOM上;如果它被销毁,它已经从DOM上移除。
    async function toggleComponent() {  
      showChild.value = !showChild.value;  
  
      // 等待DOM更新  
      await nextTick();  
  
      if (showChild.value) {  
        // 现在ChildComponent已经被创建并挂载到DOM上  
        console.log('Child component has been mounted:', childComponentRef.value);  
      } else {  
        // 现在ChildComponent已经被销毁并从DOM上移除  
        console.log('Child component has been unmounted.');  
      }  
    }  
  
    return {  
      showChild,  
      childComponentRef,  
      toggleComponent  
    };  
  }  
};  
</script>

在异步更新队列中插入自己的回调

Vue 在更新 DOM 时是异步执行的。只要侦听到数据变化,Vue 将开启一个队列,并缓冲在同一事件循环中发生的所有数据变更。如果同一个watcher被多次触发,只会被推入到队列中一次。这种在缓冲时去除重复数据,对于避免不必要的计算和 DOM 操作是非常重要的。然后,在下一个的事件循环"tick"中,Vue 刷新队列并执行实际 (已去重的) 工作。Vue 在内部对异步队列尝试使用原生的Promise.then、MutationObserver和setImmediate,如果执行环境不支持,则会采用setTimeout(fn, 0)代替。你可以通过nextTick()将回调延迟到下次 DOM 更新循环之后执行。在修改数据之后立即使用它,然后等待 DOM 更新。

处理大量数据

在处理大量数据时,Vue.js 可能会变得有点慢,因为它需要时间来处理所有的数据并更新DOM。在这种情况下,你可能想要使用nextTick()来分批处理数据,以便给浏览器一些时间来更新DOM。这可以提高应用程序的性能,并减少用户在处理大量数据时的等待时间。

javascript 复制代码
<template>  
  <div>  
    <div v-for="(item, index) in items" :key="index">{{ item }}</div>  
  </div>  
</template>  
  
<script>  
import { ref, nextTick } from 'vue';  
  
export default {  
  setup() {  
    const items = ref([]);  
  
    function fetchData() {  
      // 假设 fetchDataFromAPI 是一个异步函数,用于从 API 获取数据  
      fetchDataFromAPI().then(newItems => {  
        items.value = newItems;  
        nextTick(() => {  
          console.log('All items are rendered');  
          // 这里可以执行依赖于所有项都已渲染的代码  
        });  
      });  
    }  
  
    return {  
      items,  
      fetchData  
    };  
  }  
};  
</script>

与第三方库集成

有些第三方库可能需要直接操作DOM,或者需要在DOM更新后才能正确工作。在这种情况下,你可以使用nextTick()来确保在集成这些库时DOM已经更新。例如,你可能想要在使用一个JavaScript动画库之前等待DOM更新,以确保动画在正确的元素上运行。或者你可能想要在使用一个需要测量元素尺寸或位置的库之前等待DOM更新。在这些情况下,nextTick() 可以帮助你确保DOM已经更新并准备好与第三方库集成。

javascript 复制代码
<template>  
  <div ref="myDiv">Some Content</div>  
</template>  
  
<script>  
import { ref, onMounted, nextTick } from 'vue';  
import someLibrary from 'some-library';  
  
export default {  
  setup() {  
    const myDiv = ref(null);  
  
    onMounted(() => {  
      nextTick(() => {  
        // 确保 DOM 已经更新  
        someLibrary.doSomethingWithElement(myDiv.value);  
      });  
    });  
  
    return {  
      myDiv  
    };  
  }  
};  
</script>
相关推荐
蓝天白云下遛狗5 分钟前
goole chrome变更默认搜索引擎为百度
前端·chrome
come1123428 分钟前
Vue 响应式数据传递:ref、reactive 与 Provide/Inject 完全指南
前端·javascript·vue.js
前端风云志1 小时前
TypeScript结构化类型初探
javascript
musk12121 小时前
electron 打包太大 试试 tauri , tauri 安装打包demo
前端·electron·tauri
翻滚吧键盘2 小时前
js代码09
开发语言·javascript·ecmascript
万少2 小时前
第五款 HarmonyOS 上架作品 奇趣故事匣 来了
前端·harmonyos·客户端
OpenGL2 小时前
Android targetSdkVersion升级至35(Android15)相关问题
前端
rzl022 小时前
java web5(黑马)
java·开发语言·前端
Amy.Wang2 小时前
前端如何实现电子签名
前端·javascript·html5
海天胜景2 小时前
vue3 el-table 行筛选 设置为单选
javascript·vue.js·elementui