粥里有勺糖

vuePress-theme-reco 粥里有勺糖    2018 - 2023
粥里有勺糖 粥里有勺糖

Choose mode

  • dark
  • auto
  • light
关于我
备战春秋
  • 心得总结
  • 校招考点汇总
  • 面经汇总
  • 复习自查
技术笔记
  • 技术教程
  • 模板工程
  • 源码学习
  • 技术概念
  • 个人作品
  • 学习笔记
计算机基础
  • 算法与数据结构
  • 操作系统
  • 计算机网络
  • 设计模式
  • 剑指offer
大前端
  • javascript
  • vue
  • html
  • css
  • 🌏浏览器专题
  • Web性能优化
  • regexp
  • node
面试
  • 问解
  • javascript
  • css
  • 手撕代码
  • 性能优化
  • 综合问题
  • 面经汇总
  • 小程序
手撕代码
  • 数据结构与算法
  • javascript
  • css
个人站点
  • GitHub (opens new window)
  • 博客园 (opens new window)
  • 掘金 (opens new window)
线上作品
  • 轻取(文件收集) (opens new window)
  • 个人图床 (opens new window)
  • 考勤小程序 (opens new window)
  • 时光恋人 (opens new window)
  • 在线简历生成 (opens new window)
留言板
Github (opens new window)
author-avatar

粥里有勺糖

285

文章

40

标签

关于我
备战春秋
  • 心得总结
  • 校招考点汇总
  • 面经汇总
  • 复习自查
技术笔记
  • 技术教程
  • 模板工程
  • 源码学习
  • 技术概念
  • 个人作品
  • 学习笔记
计算机基础
  • 算法与数据结构
  • 操作系统
  • 计算机网络
  • 设计模式
  • 剑指offer
大前端
  • javascript
  • vue
  • html
  • css
  • 🌏浏览器专题
  • Web性能优化
  • regexp
  • node
面试
  • 问解
  • javascript
  • css
  • 手撕代码
  • 性能优化
  • 综合问题
  • 面经汇总
  • 小程序
手撕代码
  • 数据结构与算法
  • javascript
  • css
个人站点
  • GitHub (opens new window)
  • 博客园 (opens new window)
  • 掘金 (opens new window)
线上作品
  • 轻取(文件收集) (opens new window)
  • 个人图床 (opens new window)
  • 考勤小程序 (opens new window)
  • 时光恋人 (opens new window)
  • 在线简历生成 (opens new window)
留言板
Github (opens new window)
  • Javascript

    • 简介
    • 获取某年某月的天数
    • 日期格式化
    • 展开多级数组
    • 判断数据类型的方案
    • 文件上传与下载
    • 柯里化
    • 闭包
    • delete
    • 垃圾回收
    • 节流与防抖
    • apply,call,bind
    • blob与file
    • Event Loop
    • Promise
    • 定时器
    • 原型与原型链
    • 作用域
    • 执行上下文栈
    • 变量对象
    • 作用域链
    • ECMAScript规范解读this
    • 执行上下文
    • 参数按值传递
    • 类数组与arguments
    • 浮点数
    • Symbol的用法
    • 箭头函数
    • 类型转换
    • 获取dom元素的方式
    • 浅拷贝与深拷贝
    • ES6+的新语法糖和方法整理
    • 学习过程中学到的一些取巧之法

浅拷贝与深拷贝

vuePress-theme-reco 粥里有勺糖    2018 - 2023

浅拷贝与深拷贝

粥里有勺糖 2020-10-02 大前端javascript

# 浅拷贝与深拷贝

对象类型在赋值的过程中其实是复制了地址,从而会导致改变了一方其他也都被改变的情况,解决这种问题的方法就是利用深/浅拷贝

# 什么是浅拷贝

针对对象拷贝的场景,只拷贝了第一层,深层次(大于一层)只能拷贝到对象的引用

即深层次的对象发生了变动,会影响原来的对象与被拷贝出来的新对象

/**
 * 浅拷贝
 * @param {object} obj 
 */
function shallowCopy(obj) {
    // ...省略实现
}

const a = {
    name: 'xm', info: {
        age: 18
    }
}

const a1 = shallowCopy(a)
a1.name = 'a1'
a1.info.age = 20

console.log(a1.name === a.name) // false
console.log(a1.info.age === a.info.age) // true
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

# 局限性

浅拷贝只解决了第一层的问题,如果接下去的值中还有对象的话,拷贝的仍是引用

在大多数情况下,我们的对象的层级都是超过一层的

# 实现浅拷贝的方案

# 1. Object.assign

let a = { name:'xm' }
let a1 = Object.assign({},a)
a.name = 'a1'
console.log(a1.name === a.name) // false
1
2
3
4

# 2. 展开运算符

let a = { name: 'xm' }
let a1 = {...a}
a.name = 'a1'
console.log(a1.name === a.name) // false
1
2
3
4

针对数组还有

# 3. Array.prototype.slice

let arr = [1, 2, { name: 'xm' }]
let arr1 = Array.prototype.slice.call(arr)
1
2

# 4. Array.prototype.concat

let arr = [1, 2, { name: 'xm' }]
let arr1 = Array.prototype.concat.call(arr)
1
2

# 什么是深拷贝

  • 浅拷贝只解决了第一层的问题,如果接下去的值中还有对象的话。要解决这个问题,我们就得使用深拷贝了。

# 实现深拷贝方案

# 1. JSON.parse(JSON.stringify(object))

能应付大多数业务场景

局限:

  • 忽略undefined
  • 忽略symbol
  • 原型链上的属性无法拷贝
  • 不能处理RegExp
  • 不能正确处理Date
  • 不能序列化函数
  • 不能处理循环引用的对象

# 2. MessageChannel

  • 异步方法
  • 不能处理函数
  • 可以处理undefined与循环引用
function deepClone1(obj) {
    return new Promise(resolve => {
        let { port1, port2 } = new MessageChannel()
        port2.onmessage = ev => {
            resolve(ev.data)
        }
        port1.postMessage(obj)
    })
}
(async function () {
    let a = { a: 1, b: undefined, c: { d: 1 } }
    // 添加循环引用
    a.c.d = a.c
    let b = await deepClone1(a)
    console.log(b)
})()
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16

# 3. 手动实现一个较完整的深拷贝

戳→ 实现深拷贝 ←文章

Edit this page (opens new window)
Last Updated: 2022/5/15 12:46:34