js基础复习之自己实现一个deepClone深拷贝(详细注释版)

复习一下js基础部分的深拷贝问题:

// 这里我们先创建一个obj对象,就是我们要深拷贝的那个
var obj = {
    name: 'DeanWade',
    age: 28,
    info: {
        hobby: ['switch', 'ps4', 'basketball', {
            'coding1': 'js',
            'coding2': 'vue',
            'coding3': 'python'
        }],
        career: {
            worldwideTrading: 3,
            FrontEndEnginner: 2
        }
    }
}

// origin是拷贝目标, target是输出结果目标
function deepClone(origin, target) {
	// 如果没有指定输出结果对象,我们就默认给一个空对象{}
    var tar = target || {}
    // 把Object.prototype.toString方法拿出来保存到toStr,后面用
    toStr = Object.prototype.toString
    // 不想维护字符串, arrType就是Object.prototype.toString的结果
    // 补充知识:
    // 我们用Object.prototype.toString.call()来判断到底是对象类型还是数组类型
    // Object.prototype.toString.call([]) ===>结果是字符串[object Array]
    // Object.prototype.toString.call({}) ===>结果是字符串[object Object]
    var arrType = '[object Array]'
    
    // 遍历拷贝目标里面的每一项
    for (var k in origin) {
    	// 要判断属性一定要在目标的本身,而不是在它的__proto__上
        if(origin.hasOwnProperty(k)) {
        	// 如果origin[k]是一个对象类型并且还不是null的时候
            if(typeof origin[k] === 'object' && origin[k] !== null) {
            	// 判断origin[k]是数组还是对象
            	// 是数组我们就让目标的k属性targe[k]初始化为空数组[]
            	// 是对象我们就让标的k属性targe[k]初始化为空对象{}
                tar[k] = toStr.call(origin[k]) === arrType ? [] : {}
                // 递归deepClone, 这次递归的拷贝目标是origin[k], 输出目标是target[k]
                deepClone(origin[k], tar[k])
            // 直到一个属性不是对象的时候,就可以直接赋值不用递归循环去一层一层往里面找了
            } else {
            	// 直接赋值
                tar[k] = origin[k]
            }
        }
    }
    // 返回最终结果
    return tar
}

const newObj = deepClone(obj, {})
console.log(newObj)

运行结果:
在这里插入图片描述

这篇教程整理自小野森森老师的视频ES5-ES6『对象深拷贝』

深拷贝是指完全复制一个对象,包括它的引用类型数据(如数组、对象等),而不是只复制它们的地址或引用。深拷贝的目的是为了保持原始对象和新对象的独立性,避免它们之间相互影响。 实现深拷贝的方法有很多,比较常用的有递归、JSON序列化和反序列化、lodash库等。下面分别介绍这些方法的实现方式。 1. 递归 使用递归的方式可以遍历对象的每一个属性,并将其复制到新的对象中。同时,如果属性的值仍然是一个对象,就递归地调用深拷贝函数,直到遍历完所有的属性。 ```javascript function deepClone(obj) { if (typeof obj !== 'object' || obj === null) { return obj; } const newObj = Array.isArray(obj) ? [] : {}; for (const key in obj) { if (Object.hasOwnProperty.call(obj, key)) { newObj[key] = deepClone(obj[key]); } } return newObj; } ``` 2. JSON序列化和反序列化 使用JSON.stringify()方法将原始对象序列化为一个JSON字符串,然后使用JSON.parse()方法将JSON字符串反序列化为一个新的对象。 ```javascript function deepClone(obj) { return JSON.parse(JSON.stringify(obj)); } ``` 这种方法的缺点是,如果原始对象中有函数、正则表达式、Date等对象时,在序列化和反序列化过程中会丢失它们的类型,变成了字符串或空对象。 3. lodash库 lodash是一个常用的JavaScript工具库,提供了许多方便的函数,包括深拷贝函数。 ```javascript const _ = require('lodash'); const obj = {a: 1, b: {c: 2, d: {e: 3}}}; const newObj = _.cloneDeep(obj); ``` 使用lodash库的深拷贝函数可以处理原始对象中的函数、正则表达式、Date等对象,保持它们的类型不变。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值