Vue|数据更新,视图未更新的几种解决方案

前言:

本文主要介绍在vue项目中,遇到数据更新但视图不更新的情况时,无法使用watch监听、无法使用this.$set方法,同时使用this.$forceUpdate()无效时,所使用的解决方案。

一、问题描述

在Vue项目中,有时我们会遇到数据更新了,但是视图没有更新的情况,这是因为Vue使用的是异步更新的方式,如果数据变化时视图没有立即响应,则一般都会考虑使用以下几种解决方案:

  1. 方案一:使用watch监听对象属性变化
xml 复制代码
<template>
    <div>
        <h3>{{userInfo.name}}</h3>
        <button @click="changeUserName">点击我改变用户名</button>
    </div>
</template>

<script>
export default {
    data(){
        return {
            userInfo: {
                name: '张三',
                age: 16
            }
        }
    },
    watch:{
        'userInfo.name': {
            handler(newValue, oldValue) {
              console.log('用户名变化了,新值:' + newValue + ',旧值:' + oldValue);
            },
            immediate: true,
            deep: true
        }
    },
    methods:{
        changeUserName(){
            // 假设异步请求修改用户名
            const newUserName = '李四';
            this.userInfo.name = newUserName;
        }
    }
}
</script>
  1. 方案二:使用this.$set()强制更新响应式变量
js 复制代码
// 定义数据对象:
obj: { name: "小明", age: 18, },       

// 基本语法:
this.$set(需要改变的对象,"需要改变的对象属性","新值")

示例如下:

xml 复制代码
<template>
    <div>
        <h3>{{initList.message}}</h3>
        <button @click="changeMessage">点击我改变message值</button>
    </div>
</template>

<script>
export default {
    data(){
        return {
            initList: {
                message: 'hello'
            },
        }
    },
    methods:{
        changeMessage(){
            // 假设是异步调用获取到了新的message值
            const newMessage='world';
            this.$set(this.initList, 'message', newMessage);
        }
    }
}
</script>
  1. 方案三:使用this.$forceUpdate()强制性重新渲染
js 复制代码
methods: {
    // 在某些情况下需要强制重新渲染组件
    forceRerender() {
        // 调用 $forceUpdate()方法
        this.$forceUpdate();
    }
}

但是,在项目中,如果以上方案均无效,比如由于无法使用watchthis.$set()实现所需功能,或者使用后会使代码变得更复杂、冗余;同时层级太多或其他原因导致this.$forceUpdate()也不起作用,那么还有什么方案可以解决视图不更新的问题呢?

二、解决方案

为确保视图更新,可以先将数据赋值后再清空,最后在this.$nextTick()中处理数据,并重新赋值。代码示例如下:

js 复制代码
// 为确保视图更新,先将数据重新赋值给一个变量,然后再清空
const details = this.detailsList;
this.detailsList = [];

// 处理数据,并重新赋值
this.$nextTick(() => {
    // 处理数据
    const detailsList = this.formatList(details, data.playCourseChapterId);
    // 重新赋值
    this.detailsList = Object.assign([], detailsList);
})

这时,数据更新的同时,视图也会同步更新。

备注:

我这边的项目中,是因为层级循环太多,导致以上三种方案均不起作用。只有最后这种方案可以解决视图不更新的问题。

以上,希望对大家有帮助!

相关推荐
MageGojo2 小时前
做节日活动页时,如何用 API 快速生成对联内容
javascript·python·节日·对联生成
向上的车轮3 小时前
Next.js 入门指南:从零到一构建全栈应用
开发语言·javascript·ecmascript
freeinlife'3 小时前
精准秒表计时器实现---基于js
开发语言·前端·javascript
MaCa .BaKa3 小时前
55-宠物爱心救助领养系统-宠物救助领养系统
java·vue.js·tomcat·maven·springboot·宠物救助领养系统
优雅格子衫3 小时前
uniapp 拍照相册选取后超级好用的裁剪组件,增加水印完全自定义
开发语言·前端·javascript·uni-app·vue
AI砖家3 小时前
前端 JavaScript 异步处理全方案详解:从回调到 Observable
开发语言·前端·javascript
柒和远方4 小时前
每日一学V010: 从 Python 回到前端:一个 AI Native 开发者的 JavaScript 底层基础补全
javascript
之歆4 小时前
Day21_电商详情页核心技术实战:从LESS预处理到复杂交互实现
开发语言·前端·javascript·css·交互·less
海鸥两三4 小时前
基于 Vue 3 + 高德地图的网格规划系统实战(有源码)
前端·javascript·vue.js
逸A4 小时前
某里v2反混淆 codec 化路上踩到的两个隐蔽坑:被清零的 salt 与 opaque loop bound
javascript·人工智能·目标跟踪