• 【 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
  • 相关阅读:
    .NET Evolve 数据库版本管理工具
    Vue+ElementUi实现菜单循环
    【Flink 实战系列】Flink pipeline.operator-chaining 参数使用和解析
    vue面试题7
    Flink sql 实现 -connection-clickhouse的 source和 sink
    单元测试效率优化:为什么要对程序进行测试?测试有什么好处?
    【C++进阶】set和map的基本使用(灰常详细)
    Java正则表达式 提取文本中所有的匹配数据
    Elasticsearch 进阶
    中间件安全(概述)有中间件的各类链接和官网信息和漏洞库以及配置问题和开源工具
  • 原文地址:https://blog.csdn.net/CharlynYanyan/article/details/126649086