vue 中由浅拷贝引发问题的一些场景

news2025/1/1 22:44:20

在工作的过程中踩了很多的由浅拷贝导致的坑,今天总结在这里,希望对大家有所帮助

1. 组件中直接抛出一个引用类型变量

🌰举个例子 (ps: 以下代码为伪代码,主要展示逻辑用)

子组件(uploadImg)定义一个上传图片的组件

基础功能:a. 上传图片,上传成功后抛出change 事件。

                 b. 上传成功的图片平铺展示,并展示删除按钮。点击删除按钮,弹出二次确认框。点击确认则删除,点击取消则不删除。

// uploadImg.vue
// 上传图片组件——子组件 
<template>
   <upload @change="uploadSuccess"></upload>
   <div>
    <div v-for="(item, fileIndex) in imgUrls" :key="fileIndex" class="file-content">
          <div class="file-item">
            <div @click="delFile(fileIndex)" class="file-close">
              <el-tooltip class="item" effect="dark" content="删除" placement="right-start">
                <el-icon class="el-icon-close"></el-icon>
              </el-tooltip>
            </div>
            </div>
          </div>
        </div>
</div>
</template>
<script>
data(){
    return {
        imgUrls: []
    }
},
methods: {
    // 上传成功
    uploadSuccess (files) {
      files.forEach(item => {
        this.imgUrls.push({
          ...item,
          fileName: item.fileName
        })
      })
      this.$emit('change', this.imgUrls)
    },
    // 删除文件
    delFile (fileIndex) {
      this.$confirm('图片删除后不能恢复,确认删除?', '提示').then(() => {
        this.imgUrls.splice(fileIndex, 1)
      })
      this.$emit('change', this.imgUrls)  
    }
}
</script>

父组件使用方法1

<template>
    <upload-img @change="change"></upload-img>
</template>
<script>
    data(){
        return {
            imageList
        }
    },
    methods: {
        change(list) {
              this.imageList = list
              console.log(this.imageList)
        }
    }
</script>

我们使用以上方法,点击删除图片,确认删除。

获取到的 this.imageList,是删除成功之后的图片数组,无异常。删除和上传给父组件的值都是正确的。

父组件使用方法2

<template>
    <upload-img @change="change"></upload-img>
</template>
<script>
    data(){
        return {
            imageList
        }
    },
    methods: {
        change(list) {
              this.imageList = list.map(item => item.url)
              console.log(this.imageList)
        }
    }
</script>

我们使用以上方法,点击删除图片,确认删除。

获取到的 this.imageList,是删除之前的图片数组,有异常。

为啥会产生这样的异常呐??我们来分析一下

 由上图可见,抛出change事件的时机,和弹出二次确认弹窗的时间是异步的。

即:在弹出二次确认弹窗的时候,chang事件就已经触发了。反而在点击二次确认弹窗的确定按钮的时候并未触发chang事件。

那么问题来了,为啥第一种方式无异常咧?因为我们change事件抛出的是一个引用类型的数据,并且父组件对该值进行了浅拷贝。等子组件的imgUrls值更新之后,父组件的imageList的值也进行了更新。

第二种方式由异常的原因:第二种方法在赋值之前使用了一个map 的方法,map 会返回一个新数组,即进行了深拷贝。

2. 直接将引用类型赋值给一个新的变量,对新变量进行处理

🌰举个例子

form表单中,输入一些数据,调用后端接口传参数是,需要对数据进行一些处理,比如金额,需要由元转成分

这时直接进行转化,就是导致页面上的展示数据,也会进行相应的变动。

常见于:数据长度变多或者变少,对象修改了某一个key 的值,却无变化。

常见的浅拷贝的方法?

1. Object.assign()

2. 解构赋值

3. ‘=’赋值

常见解决方法

1. 手写深拷贝递归复制

2. JSON做字符串转换

var obj1 = { body: { a: 10 } };
var obj2 = JSON.parse(JSON.stringify(obj1))

该方法有副作用

        a. 它会抛弃对象的constructor。也就是深拷贝之后,不管这个对象原来的构造函数是什么,在深拷贝之后都会变成Object。

        b.RegExp对象是无法通过这种方式深拷贝

        c.function没办法转成JSON

        

3. 使用Object.create()方法

直接使用var newObj = Object.create(oldObj),可以达到深拷贝的效果。

附上创建对象的一些方法的差异:

附上创建对象的一些方法的差异:Object.create()、new Object()和{}的区别 - 掘金

4. 三方函数(常用)

如lodash

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

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

相关文章

线 程 同 步、线程的死锁问题

线程同步&#xff1a; 模拟售票程序出现问题&#xff1a;当多个线程同时访问共享数据时&#xff0c;产生无序、重复、超额售票等多线程安全问题 解决&#xff1a;将多个线程需要访问的共享数据&#xff0c;包装起来视为一个整体&#xff0c;确保一次只有一个线程执行流访问共享…

春节福利丨神策数据 2022 年数字化营销资料打包全送

2022 年&#xff0c;神策数据出品多份行业研究报告&#xff0c;覆盖银行、证券、零售、教育、电商、融合媒体等多个行业&#xff0c;帮助更多企业通过多视角洞见紧握数字化营销的方向和趋势&#xff0c;用方法论结合落地实践驱动企业数字化经营。01B2B 电商数字化运营聚焦四类 …

【自学Python】Python查找字符串位置

Python查找字符串位置 大纲 Python查找字符串位置教程 在开发过程中&#xff0c;很多时候我们有在一个 字符串 中查找另一个字符串位置的需求&#xff0c;在 Python 中&#xff0c;在一个字符串中查找另一个字符串的位置我们使用 index() 函数。 index() 函数的功能与 find(…

力扣(78.90)补9.22

78.子集 感觉不太难&#xff0c;但是就是不会写。感觉回溯里有很多细节问题。 class Solution { private: vector<vector<int>> res; vector<int> num; void back(vector<int>& nums,int index,int end){ res.push_back(num); …

【JavaEE初阶】第三节.多线程基础篇

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 目录 文章目录 前言 一、认识线程 二、多线程程序 2.1 第一个Java多线程程序 2.2 怎么样观察线程的详细情况 2.3 sleep方法 2.4 run 和 start 方法的区别是什么 三、创…

在springboot中配置热部署

今天什么节日也不是&#xff0c;那就祝大家今天快乐。 热部署 所谓热部署&#xff0c;就是在应用正在运行的时候升级软件&#xff0c;却不需要重新启动应用。对于Java应用程序来说&#xff0c;热部署就是在运行时更新Java类文件。在以往&#xff0c;我们对java代码进行修改之后…

【HBU】大一下期末重点

物理简答题一、牛顿第二定律&#xff08;Fdp/dt &#xff09;&#xff1a;1. 飞机怕小鸟:Fdp/dt 力与作用时间的乘积等于物体动量的变化。由于小鸟与飞机相对速度很大&#xff0c;作用时间很短,产生的作用力很大&#xff0c;当小鸟与飞机向撞&#xff0c;效果与同质量炮弹撞飞机…

正则表达式入门及常用正则表达式

常用正则表达式 1 正则表达式的基础概念 1.1 预定义字符 . 表示任何字符&#xff08;与行结束符可能匹配也可能不匹配&#xff09; \d 数字&#xff1a;[0-9] \D 非数字字符&#xff1a;[^0-9] \s 空白字符&#xff1a;[\t\n\xoB\f\r] \S 非空白字符&#xff1a;[^\s] \w 单…

k8s的YAML部署rocketmq记录

说明 测试环境是k8s集群&#xff0c;在上边部署一套单节点的rocketmq nameserver部署 Service和StatefulSet脚本如下 apiVersion: v1 kind: Service metadata:labels:app: rocketmqnamesrvname: rocketmqnamesrv spec:type: ClusterIPports:- port: 9876targetPort: 9876na…

Pytorch深度学习【十四】

批量归一化 归一化 损失出现在最后&#xff0c;后面的层(高级语义层)训练较快数据输入在最底部 底部的层训练慢底部层一变化&#xff0c;所有高级语义层都得跟着变最后的那些层需要重新学习多次—收敛速度变慢 问题—是否可以在学习底部层的时候避免变化顶部层 批量归一化 固定…

8 个精彩的免费 G​​IS 软件资源分享

有人说&#xff1a;一个人从1岁活到80岁很平凡&#xff0c;但如果从80岁倒着活&#xff0c;那么一半以上的人都可能不凡。 生活没有捷径&#xff0c;我们踩过的坑都成为了生活的经验&#xff0c;这些经验越早知道&#xff0c;你要走的弯路就会越少。 GIS 软件有两种通用格式&a…

【面试题】2023年前端最新面试题-性能优化篇

原文见&#xff1a;语雀&#xff08;https://www.yuque.com/deepstates/interview/xtt59x&#xff09; ● 性能指标 ● 分析工具 ● 优化方式 ○ 加载 ○ 渲染 ● 专题优化 ○ 技术栈&#xff1a;react ○ 浏览器 ○ 打包工具&#xff1a;webpack ● 项目 ⭐️⭐️⭐️ 相关知…

如何隐藏电脑硬盘分区?

无论是个人还是公司的电脑我们都会储存一些重要的数据&#xff0c;有些甚至还是涉及个人隐私或公司的商业机密。为了更好地保护电脑磁盘中的重要资料&#xff0c;部分用户希望能将硬盘分区隐藏起来。那么怎么隐藏硬盘分区呢&#xff1f;方法一&#xff1a;使用磁盘管理隐藏硬盘…

将vscode打造为你的开发工具的首选

文章目录前言vscode主要配置vscode的两个主要快捷键Java配置JDK和Gradle环境主要插件常见的配置launch.json配置运行测试用例常见问题Python主要插件settings.json配置Javascript/typescript常用插件settings.json样例Golang参考前言 什么是IDE? IDE 文本编辑 搜索 代码导…

Matlab矩阵和数组的操作

一、矩阵的建立 1、直接输入法 将矩阵的元素用方括号括起来&#xff0c;按矩阵行的顺序输入各元素&#xff0c;同一行的各元素之间用空格或逗号分隔&#xff0c;不同行的元素之间用分号分隔。 A [16 3 2 13; 5 10 11 8; … 9 6 7 12; 4 15 14 1] A 16 3 2 13 5 10 11 8 9 6…

如何有效的增加 shopee 的流量?

很多卖家选择在跨境电商平台开店。说到跨境电商&#xff0c;大家首先想到的应该是亚马逊、易趣等电商平台&#xff0c;边肖会在shopee平台上给大家带来店铺。新店如何获得流量&#xff1f;有哪些方式&#xff1f;米贸搜为你整理如下&#xff1a;shopee店铺如何获取流量&#xf…

Python学习笔记——函数

函数是组织好的&#xff0c;可重复使用的&#xff0c;用来实现单一&#xff0c;或相关联功能的代码段。能提高应用的模块性&#xff0c;和代码的重复利用率。定义函数定义函数使用关键字def&#xff0c;后接函数名&#xff0c;再后接放在圆括号&#xff08;&#xff09;中的可选…

P1036 [NOIP2002 普及组] 选数————C++

题目 [NOIP2002 普及组] 选数 题目描述 已知 nnn 个整数 x1,x2,⋯,xnx_1,x_2,\cdots,x_nx1​,x2​,⋯,xn​&#xff0c;以及 111 个整数 kkk&#xff08;k<nk<nk<n&#xff09;。从 nnn 个整数中任选 kkk 个整数相加&#xff0c;可分别得到一系列的和。例如当 n4n4…

支持ITIL的IT帮助台

什么是ITIL 信息技术基础架构库 &#xff08;ITIL&#xff09; 是一套集成的最佳实践&#xff0c;旨在帮助企业向客户提供 IT 服务。ITIL是由中央设计的广泛采用的框架 计算机和电信局&#xff08;CCTA&#xff09;&#xff0c;英国的政府机构;它目前由AXELOS Ltd拥有。 ITIL…

【微信小程序】实用教程03-自定义底部导航(含自定义tabBar导航高亮需点击两次的解决方案)

开始前&#xff0c;请先完成底部导航的开发&#xff0c;详见 【微信小程序】实用教程01-注册登录账号&#xff0c;获取 AppID、下载安装开发工具、创建项目、上传体验 显然&#xff0c;纯文字的底部导航有点low&#xff0c;还是需要有图标的才酷&#xff0c;下面我们一起来实…