解决el-table数据变化,视图不更新的问题

今天在项目上要实现一个在表格里使用el-radio实现单选功能,遇到一个问题是选择不同行内的单选框时,其他行的radio状态不更新,如图:(左侧是正常效果,右侧为bug图)

查找原因,发现在script中table的:data数据变量已经更新,但视图没跟着及时渲染。

(如图,第三行的checked是没有值的,但el-radio仍是选中状态)

原理是在改变数组或对象的的某个属性时,diff算法并没有检测到数据变化,所以不会更新视图,这时需要使用JSON.parse(JSON.stringify(newArray))对数据进行一层深拷贝,即可:

<el-table-column label="位置" width="180" >
  <template #default="scope">
     <div v-show="scope.row.checked || scope.$index === moveIndex">
        <el-radio-group 
          @change="handleChange(scope.row, scope.$index, scope.row.checked)" 
          v-model="scope.row.checked"
        >
            <el-radio label="-1" size="large">上</el-radio>
            <el-radio label="1" size="large">下</el-radio>
        </el-radio-group>
     </div>
  </template>
</el-table-column>
<script>
export default {
    data(){
        return {
            data:[]
        }
    },
    methods:{
        async handleChange(row, index, checked){
          let newArray = await this.data.map((item,idx)=>{
            if(item == row){
              item.checked = checked;
            }else{
              item.checked = "";
            }
            return item;
          })
          // 视图更新的关键步骤!!
          this.data = JSON.parse(JSON.stringify(newArray));
        }
    }
}
</script>

此前,我尝试使用了this.$nextTick()方式,不生效,如果数据更新是异步进行的(例如通过API请求),确保更新数据后,使用Vue的nextTick方法来等待DOM更新。

<script>
export default {
    data(){
        return {
            data:[]
        }
    },
    methods:{
        handleChange(row, index, checked){
            this.$nextTick(function(){
                this.data.forEach((item)=>{
                    if(item == row){
                        item.checked = checked;
                    }else{
                        item.checked = ""
                    }
                })
            })
        }
    }
}
</script>

在网上查找类似问题,还尝试使用了this.$set()方法,这个方式看介绍是对数据进行响应式,就是解决直接改动数组并不能驱动视图更新的问题,但还是不生效,不知道怎么回事?

export default {
    data(){
        return {
            data:[]
        }
    },
    methods:{
        handleChange(row, index, checked){
                this.data.forEach((item,idx)=>{
                    if(item == row){
                        item.checked = checked;
                        //特意使用了$set(),还是不好使
                        this.$set(this.data,idx,item)
                    }else{
                        item.checked = ""
                        this.$set(this.data,idx,item)
                    }
                })
        }
    }
}

希望能解决你遇到的同类问题,喜欢的可以点赞收藏~

  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
如果你使用的是 Element UI 的 `el-table` 组件,并且发现在对数据进行增删改操作后,视图没有自动更新,可以尝试以下方法: 1. 使用 `this.$set` 方法更新数据 在 Vue 中,如果你对一个数组或对象进行了增删改操作,Vue 是无法检测到这个操作的。因此,需要使用 `this.$set` 方法来手动更新数据。 例如,如果你要更新 `tableData` 数组中的第一项数据: ```javascript this.$set(this.tableData, 0, { name: 'new name', age: 20 }); ``` 这个方法会告诉 Vue,你修改了 `tableData` 数组中的第一项数据,需要重新渲染视图。如果你直接修改了这个数组中的某一项数据,Vue 是无法检测到这个修改的。 2. 在 `el-table-column` 组件上使用 `prop` 属性 如果你使用了 `el-table-column` 组件来渲染表格列,需要在组件上使用 `prop` 属性,指定渲染的数据字段。 例如: ```html <el-table-column label="姓名" prop="name"></el-table-column> <el-table-column label="年龄" prop="age"></el-table-column> ``` 这样,当你对 `tableData` 数组中的数据进行增删改操作后,视图就会自动更新。这是因为 `el-table-column` 组件会监听 `prop` 属性对应的数据字段,一旦数据发生变化,就会重新渲染视图。 以上两种方法,可以分别解决不同的场景下 `el-table` 视图更新问题。如果以上两种方法都无法解决你的问题,可以提供更详细的代码和数据,以便更好地定位问题

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值