• 【 Javascript 】Object.assign( )


    Object.assign

    Object(对象):基础知识

    克隆与合并,Object.assign

    那么,拷贝一个对象变量会又创建一个对相同对象的引用。

    但是,如果我们想要复制一个对象,那该怎么做呢?

    我们可以创建一个新对象,通过遍历已有对象的属性,并在原始类型值的层面复制它们,以实现对已有对象结构的复制。

    就像这样:

    let user = {
      name: "John",
      age: 30
    };
    
    let clone = {}; // 新的空对象
    
    // 将 user 中所有的属性拷贝到其中
    for (let key in user) {
      clone[key] = user[key];
    }
    
    // 现在 clone 是带有相同内容的完全独立的对象
    clone.name = "Pete"; // 改变了其中的数据
    
    alert( user.name ); // 原来的对象中的 name 属性依然是 John
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16

    我们也可以使用 Object.assign 方法来达成同样的效果。

    语法是:

    Object.assign(dest, [src1, src2, src3...])
    
    • 1
    • 第一个参数 dest 是指目标对象。
    • 更后面的参数 src1, ..., srcN(可按需传递多个参数)是源对象。
    • 该方法将所有源对象的属性拷贝到目标对象 dest 中。换句话说,从第二个开始的所有参数的属性都被拷贝到第一个参数的对象中。
    • 调用结果返回 dest

    例如,我们可以用它来合并多个对象:

    let user = { name: "John" };
    
    let permissions1 = { canView: true };
    let permissions2 = { canEdit: true };
    
    // 将 permissions1 和 permissions2 中的所有属性都拷贝到 user 中
    Object.assign(user, permissions1, permissions2);
    
    // 现在 user = { name: "John", canView: true, canEdit: true }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    如果被拷贝的属性的属性名已经存在,那么它会被覆盖:

    let user = { name: "John" };
    
    Object.assign(user, { name: "Pete" });
    
    alert(user.name); // 现在 user = { name: "Pete" }
    
    • 1
    • 2
    • 3
    • 4
    • 5

    我们也可以用 Object.assign 代替 for..in 循环来进行简单克隆:

    let user = {
      name: "John",
      age: 30
    };
    
    let clone = Object.assign({}, user);
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    它将 user 中的所有属性拷贝到了一个空对象中,并返回这个新的对象。

    还有其他克隆对象的方法,例如使用 spread 语法 clone = {...user},在后面的章节中我们会讲到。

    深层克隆

    到现在为止,我们都假设 user 的所有属性均为原始类型。但属性可以是对其他对象的引用。

    例如:

    let user = {
      name: "John",
      sizes: {
        height: 182,
        width: 50
      }
    };
    
    alert( user.sizes.height ); // 182
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    现在这样拷贝 clone.sizes = user.sizes 已经不足够了,因为 user.sizes 是个对象,它会以引用形式被拷贝。因此 cloneuser 会共用一个 sizes:

    let user = {
      name: "John",
      sizes: {
        height: 182,
        width: 50
      }
    };
    
    let clone = Object.assign({}, user);
    
    alert( user.sizes === clone.sizes ); // true,同一个对象
    
    // user 和 clone 分享同一个 sizes
    user.sizes.width++;       // 通过其中一个改变属性值
    alert(clone.sizes.width); // 51,能从另外一个获取到变更后的结果
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15

    为了解决这个问题,并让 userclone 成为两个真正独立的对象,我们应该使用一个拷贝循环来检查 user[key] 的每个值,如果它是一个对象,那么也复制它的结构。这就是所谓的“深拷贝”。

    我们可以使用递归来实现它。或者为了不重复造轮子,采用现有的实现,例如 lodash 库的 _.cloneDeep(obj)

    使用 const 声明的对象也是可以被修改的

    通过引用对对象进行存储的一个重要的副作用是声明为 const 的对象 可以 被修改。

    例如:

    const user = {
      name: "John"
    };
    
    user.name = "Pete"; // (*)
    
    alert(user.name); // Pete
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    看起来 (*) 行的代码会触发一个错误,但实际并没有。user 的值是一个常量,它必须始终引用同一个对象,但该对象的属性可以被自由修改。

    换句话说,只有当我们尝试将 user=... 作为一个整体进行赋值时,const user 才会报错。

    也就是说,如果我们真的需要创建常量对象属性,也是可以的,但使用的是完全不同的方法。我们将在 属性标志和属性描述符 一章中学习它。

    总结

    对象通过引用被赋值和拷贝。换句话说,一个变量存储的不是“对象的值”,而是一个对值的“引用”(内存地址)。因此,拷贝此类变量或将其作为函数参数传递时,所拷贝的是引用,而不是对象本身。

    所有通过被拷贝的引用的操作(如添加、删除属性)都作用在同一个对象上。

    为了创建“真正的拷贝”(一个克隆),我们可以使用 Object.assign 来做所谓的“浅拷贝”(嵌套对象被通过引用进行拷贝)或者使用“深拷贝”函数,例如 _.cloneDeep(obj)

    _.cloneDeep(value)

    source npm package

    This method is like _.clone except that it recursively clones value.

    Since

    1.0.0

    Arguments
    1. value (*): The value to recursively clone.
    Returns

    (*): Returns the deep cloned value.

    Example
    var objects = [{ 'a': 1 }, { 'b': 2 }];
     
    var deep = _.cloneDeep(objects);
    console.log(deep[0] === objects[0]);
    // => false
    
    • 1
    • 2
    • 3
    • 4
    • 5
  • 相关阅读:
    SpringBoot使用云服务器实现文件上传和下载
    手把手教你搭建农产品商城小程序:详细步骤解析
    如何使用 CocoaPods 将智能生活 App SDK iOS 版快速集成到项目中
    基于 Vue2 组件之间的通信
    HttpStatusCodeException.getResponseBodyAsString 乱码
    【线性系统理论】笔记三
    电商公司如何使用IP地址归属地?
    【SemiDrive源码分析】【驱动BringUp】42 - Mailbox Demo实现
    论文超详细精读|五千字:STGR
    Python import module package 相关
  • 原文地址:https://blog.csdn.net/CharlynYanyan/article/details/126649086