聊聊vue的nextTick方法

news2024/12/26 2:43:36

前言

nextTick是面试常考的vue中的一个重要知识点,但是很多小伙伴常常无法真正的理解nextTick的执行机制,并且背后包含的许多vue的重要知识。本文会把nextTick聊的非常细,让大家彻底搞懂它。

正文

那么在nextTick正式登场之前呢,我们先看一个例子:

页面上有20个li元素,当点击更新列表按钮时,页面会增加10个里li元素。假如现在为了提供用户更好的体验,将页面滚动到新添加元素中最后一个元素的位置,scrollIntoView 原生JS自带的方法,滚动到指定的dom结构上去,behavior: 'smooth' 指定什么类型的滚动行为。 那么该如何去书写?下面的写法可以吗?

<template>
    <div>
        <button @click="updateList">更新列表</button>
        <ul>
            <li v-for="n in list">{{ n }}</li>
        </ul>
    </div>
</template>

<script setup>
import { ref } from 'vue';

const list = ref(new Array(20).fill(0))

const updateList = () => {
    //解构是因为我们要一个一个元素push进去,而不是整个数组
    list.value.push(...(new Array(10).fill(1)))
    const liItem = document.querySelector('li:last-child')
    liItem.scrollIntoView({ behavior: 'smooth' })
}
</script>

<style lang="css" scoped>
li{
    height: 100px;
    background-color: #21e07a;
    margin: 10px;
}
</style>

很明显,页面滚动到一半就停止了,并没有滚到最后一个元素处,这是为什么呢? 这里我们可以复习一下vue中的生命周期的相关知识。

在vue组件渲染的整个过程过程中,setup语法糖也就是组合式api中的逻辑最先执行,其次才是编译模板、挂载组件、渲染DOM节点。那么在这个例子中,当点击按钮后,执行点击事件的回调函数中逻辑是最先执行的,然后添加li元素,再去获取页面更新后的最后一个DOM节点。但是页面更新DOM节点是需要时间的,执行document.querySelector('li:last-child')时能获取到哪取决于电脑的性能,所以在添加多个DOM节点后很可能无法滚动到最后一个节点就会停止。

那么怎么解决呢?

没错,用nextTick来解决。

nextTick是什么

是vue内置的一个api,在页面渲染完成后执行回调函数中的逻辑。接受一个回调函数作为参数。nextTick 是异步操作,且属于微任务。 所以会比 类似于setTimeout这样的延迟函数 更快执行。

那么很显然,这个场景非常适合用nextTick(仅展示JS部分):

<script setup>
import { nextTick, ref } from 'vue';
import { myNextTick } from './next-tick';

const list = ref(new Array(20).fill(0))

const updateList = () => {
    //解构是因为我们要一个一个元素push进去,而不是整个数组
    list.value.push(...(new Array(10).fill(1)))

    nextTick(() => { //保证浏览器更新dom完毕后
        const liItem = document.querySelector('li:last-child')
        // scrollIntoView 原生JS自带的方法,滚动到指定的dom结构上去
        // behavior: 'smooth' 指定什么类型的滚动行为
        liItem.scrollIntoView({ behavior: 'smooth' })
    })
}
</script>
怎么样,是不是顺畅多了!

手写nextTick

nextTick的手写也相对比较简单,那么下面是实现代码:

function myNextTick(fn) {
    let app = document.getElementById('app')

    // 使用MutationObserver方法监听dom
    var observerOptions = {
        childList: true, // 观察目标子节点的变化,是否有添加或者删除
        attributes: true, // 观察属性变动
        subtree: true, // 观察后代节点,默认为 false
      };
    // 要保证fn在dom更新完成后再调用
    // 创建一个DOM监听器    
    let observer = new MutationObserver((el,obs) => {
        //当被监听的DOM节点更新完成时,该回调会触发
        fn()
    })
    //将目标节点和callback绑定,并定义observerOptions来配置需要监听dom的哪些变化
    observer.observe(app,observerOptions)
}

传入的fn是需要执行的回调函数。那么如何保证fn在页面渲染完成后才调用呢?

这里使用了JS原生的MutationObserver方法监视DOM 树。MutationObserver方法接受回调函数(callback)作为参数。

这里关键使用了MutationObserver构造函数身上的observe方法,去配置 MutationObserver 在 DOM 更改匹配给定选项时,通过其回调函数开始接收通知,这里的给定选项就是observerOptions对象中的属性。当 observer 发现匹配观察请求中指定的配置项的更改时,callback() 方法便会被调用。所以只要将fn放入callback中调用即可。

最后

欢迎在评论区留言!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1517416.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

深度解析Elasticsearch索引数据量过大的优化与部署策略

✨✨谢谢大家捧场&#xff0c;祝屏幕前的小伙伴们每天都有好运相伴左右&#xff0c;一定要天天开心哦&#xff01;✨✨ &#x1f388;&#x1f388;作者主页&#xff1a; 喔的嘛呀&#x1f388;&#x1f388; 目录 引言 1. 分片和副本策略 1.1分片策略 1.1.1 数据量 1.1.…

Gitee 服务器

Git 服务器集成 1. 创建仓库 2. 远程仓库简易操作指令 # Git 全局设置&#xff0c;修改成自己的信息 git config --global user.name "Muko" git config --global user.email "txk0x7d2163.com" # 创建 git 仓库&#xff0c;基本操作指令和其他远程仓库一…

边缘计算与物联网的核心 —— 低功耗芯片

一、低功耗芯片 在边缘计算与物联网&#xff08;IoT&#xff09;中&#xff0c;低功耗芯片扮演了至关重要的角色&#xff0c;主要体现在以下几个方面&#xff1a; 延长设备寿命&#xff1a;物联网设备通常需要部署在难以更换电池或不方便进行频繁维护的环境中&#xff0c;比如…

「Paraverse平行云」受邀参与编写国内首个3D数字内容生产技术白皮书

1月26日&#xff0c;2024中关村论坛系列活动——数据&#xff08;数字&#xff09;资产高质量发展大会在银保国际会议中心成功举办。本届数据&#xff08;数字&#xff09;资产高质量发展大会由市科委中关村管委会、市经济和信息化局、石景山区人民政府主办&#xff0c;3D/XR产…

【算法与数据结构】深入解析二叉树(一)

文章目录 &#x1f4dd;数概念及结构&#x1f320; 树的概念&#x1f309;树的表示&#x1f320; 树在实际中的运用&#xff08;表示文件系统的目录树结构&#xff09; &#x1f309;二叉树概念及结构&#x1f320;概念&#x1f309;数据结构中的二叉树&#x1f320;特殊的二叉…

uniapp实现页面固定区域转为base64图片预览并手动保存本地

uniapp实现页面固定区域转为base64图片预览并手动保存本地 声明&#xff1a;H5目前没有实现直接长按保存到手机图库&#xff08;浏览器可以直接保存为图片&#xff09;&#xff0c;所以将过程中得到的base64转为真实图片地址用a标签click的方式实现预览&#xff0c;并手动长按…

微服务分布式springcloud的体育场地预约系统演kdm1z

体育场馆设施预约系统是在实际应用和软件工程的开发原理之上&#xff0c;运用java语言以及Springcloud框架进行开发。首先要进行需求分析&#xff0c;分析出体育场馆设施预约系统的主要功能&#xff0c;然后设计了系统结构。整体设计包括系统的功能、系统总体结构、系统数据结构…

node.js入门—day02

个人名片&#xff1a; &#x1f60a;作者简介&#xff1a;一名大二在校生 &#x1f921; 个人主页&#xff1a;坠入暮云间x &#x1f43c;座右铭&#xff1a;给自己一个梦想&#xff0c;给世界一个惊喜。 &#x1f385;**学习目标: 坚持每一次的学习打卡 文章目录 什么是单线程…

NVIDIA vGPU三种授权方式(个人玩家版)

NVIDIA vGPU三种授权方式(个人玩家版) 旧版本的License Server搭建(比较推荐)说明搭建所需文件创建一个Linux虚拟机(我创建的是Ubuntu 18.04.06)修改虚拟机的MAC地址关闭虚拟机的时间同步及修改系统时间安装java安装Apache Tomcat安装许可证服务器软件上传授权文件新版本…

HarmonyOS NEXT应用开发—翻页动效案例

介绍 翻页动效是应用开发中常见的动效场景&#xff0c;常见的有书籍翻页&#xff0c;日历翻页等。本例将介绍如何通过ArkUI提供的显示动画接口animateTo实现翻页的效果。 效果图预览 使用说明 本例通过setInterval函数每秒调用一次翻页动画&#xff0c;实现连续翻页效果。 …

【OpenCV】手写字符分割

OpenCV 是一个开源的计算机视觉&#xff08;Computer Vision&#xff09;与机器学习软件库&#xff0c;提供了多种图像处理算法与接口。在 OCR 技术中&#xff0c;字符分割用于提取图像中的文字信息&#xff0c;可以应用于车牌识别、身份证识别、文档扫描等场景。本文主要记录如…

基于单片机的家庭防盗报警系统

摘 要 随着社会的发展&#xff0c;人们生活水平的不断提高和家居用品的高档化&#xff0c;家庭安全隐患也随之增加&#xff0c;所以人们便进一步提高了对家庭的保护意识。因此&#xff0c;这就不得不促使安全防盗报警系统的普及与推广。 然而传统的防盗措施难以实现人们的需求…

Tomcat下载安装及纯手动发布一个应用

文章目录 javaWeb介绍一. 下载tomcat二、部署Web项目准备三. 验证tomcat配置是否成功四、安装包中各个文件的解释与用途五、纯手动部署web项目 javaWeb介绍 1、什么是JavaWeb&#xff1f; JavaWeb是一种使用Java语言编写的基于Web的应用程序开发技术。它是通过Java的Web开发框…

Web框架开发-Django的视图层

一、视图函数 一个视图函数,简称视图,是一个简单的Python 函数,它接受Web请求并且返回Web响应。响应可以是一张网页的HTML内容,一个重定向,一个404错误,一个XML文档,或者一张图片. . . 是任何东西都可以。无论视图本身包含什么逻辑,都要返回响应。代码写在哪里也无所谓…

某狗网翻译接口逆向之webpack扣取

​​​​​逆向网址 aHR0cHM6Ly9mYW55aS5zb2dvdS5jb20 逆向链接 aHR0cHM6Ly9mYW55aS5zb2dvdS5jb20vdGV4dA 逆向接口 aHR0cHM6Ly9mYW55aS5zb2dvdS5jb20vYXBpL3RyYW5zcGMvdGV4dC9yZXN1bHQ 逆向过程 请求方式&#xff1a;POST 参数构成&#xff1a; 【s】 1b921dbefaa8d939afca…

网络编程套接字(4)——Java套接字(TCP协议)

目录 一、Java流套接字通信模型 二、TCP流套接字编程 1、ServerSocket ServerSocket构造方法&#xff1a; ServerSocket方法: 2、Socket Socket构造方法&#xff1a; Socket方法&#xff1a; 三、代码示例&#xff1a;回显服务器 1、服务器代码 代码解析 2、客户端…

谁将主导未来AI市场?Claude3、Gemini、Sora与GPT-4的技术比拼

【最新增加Claude3、Gemini、Sora、GPTs讲解及AI领域中的集中大模型的最新技术】 2023年随着OpenAI开发者大会的召开&#xff0c;最重磅更新当属GPTs&#xff0c;多模态API&#xff0c;未来自定义专属的GPT。微软创始人比尔盖茨称ChatGPT的出现有着重大历史意义&#xff0c;不亚…

基于word2vec 和 fast-pytorch-kmeans 的文本聚类实现,利用GPU加速提高聚类速度

文章目录 简介GPU加速 代码实现kmeans聚类结果kmeans 绘图函数相关资料参考 简介 本文使用text2vec模型&#xff0c;把文本转成向量。使用text2vec提供的训练好的模型权重进行文本编码&#xff0c;不重新训练word2vec模型。 直接用训练好的模型权重&#xff0c;方便又快捷 完整…

软件无线电系列——模拟无线电、数字无线电、软件无线电

本节目录 一、模拟无线电 二、数字无线电 1、窄带数字无线电 2、宽带数字无线电 三、软件无线电本节内容 一、模拟无线电 20世纪80年代的模拟体制(美国的AMPS/欧洲的TACS)被称为第一代移动通信&#xff0c;简称1G,主要目标是为在大范围内有限的用户提供移动电话服务。最主要的…

uniapp遇到的问题

【uniapp】小程序中input输入框的placeholder-class不生效解决办法 解决&#xff1a;写在scope外面 uniapp设置底部导航 引用&#xff1a;https://www.jianshu.com/p/738dd51a0162 【微信小程序】moveable-view / moveable-area的使用 https://blog.csdn.net/qq_36901092/…