博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
JavaScript深拷贝、浅拷贝
阅读量:6694 次
发布时间:2019-06-25

本文共 3062 字,大约阅读时间需要 10 分钟。

JavaScript深拷贝、浅拷贝

浅拷贝:浅拷贝只是复制了内存地址,如果原地址中的对象改变了,浅拷贝出来的对象也会相应改变。

深拷贝:开辟了一块新的内存存放地址和地址指向的对象,原地址的任何对象改变了,深拷贝出来的对象不变。

浅拷贝数组(只拷贝第一级数组):

1.直接遍历
var arr = [1,2,3,4];function copy(arg){    var newArr = [];  for(var i = 0; i < arr.length; i++) {    newArr.push(arr[i]);  }  return newArr;}var newArry = copy(arr);console.log(newArry);newArry[0] = 10;console.log(newArry); // [10,2,3,4]console.log(arr)  // [1,2,3,4]复制代码
2.slice
var arr = [1,2,3,4]var copyArr = arr.slice();copyArr[0] = 10;console.log(copyArr); // [10,2,3,4]console.log(arr); // [1,2,3,4]复制代码

slice(start,end)slice()方法返回一个数组中复制出来的元素组成新数组,start指起始元素下标,end指终止元素下标

slice()不带任何参数时,默认返回一个和原数组一样的新数组

3.concat()
var arr = [1,2,3,4]var copyArr = arr.concat();copyArr[0] = 10;console.log(copyArr); // [10,2,3,4]console.log(arr); // [1,2,3,4]复制代码

array.concat(array1,array2,.......,arrayN)concat()方法用于连接两个或多个数组(不会改变原数组,返回被连接数组的副本)

然而如果第一级数组元素是对象或数组,上面三种方式都失效:
var arr = [  {
number:1}, {
number:2}, {
number:3} ]var copyArr = arr.slice();copyArr[0].number = 10;console.log(copyArr); // [{number: 100}, { number: 2 },{ number: 3 }]console.log(arr); // [{number: 100}, { number: 2 }, { number: 3 }]复制代码

浅拷贝对象(如果对象中的值不为数组或对象)

1.直接遍历
var obj = {    name: "张三",    job: "学生"  }    function copy (arg) {    let newobj = {}    for(let item in arg) {      newobj[item] = arg[item];    }    return newobj;  }    var copyobj = copy(obj)  copyobj.name = "李四"  console.log(copyobj) // {name: '李四', job:: '学生'}  console.log(obj) // {name: '张三', job:: '学生'}复制代码
2.ES6的Object.assign
var obj = {  name: '张三',  job: '学生'}var copyobj = Object.assign({},obj)copyobj.name = '李四'console.log(copyobj) // {name: '李四', job:: '学生'}console.log(obj)    // {name: '张三', job:: '学生'}复制代码

Object.assign:用于对象的合并,将源对象(source)的所有可枚举属性,复制到目标对象(target),并返回合并后的target

用法Object.assign(target, source1, source2); 所以 copyObj = Object.assign({}, obj); 这段代码将会把obj中的一级属性都拷贝到 {}中,然后将其返回赋给copyObj

3.ES6扩展运算符

var obj = {  name: '张三',  job: '学生'}var copyobj = {...obj}copyobj.name = '李四'console.log(copyobj)console.log(obj)复制代码

扩展运算符(...)用于取出参数对象的所有可遍历属性,拷贝到当前对象之中

深拷贝

JSON.stringify()JSON.parse()

JSON.stringify把对象转成字符串,再用JSON.parse把字符串转成新的对象。

但是这种方法也有不少坏处,譬如它会抛弃对象的constructor。也就是深拷贝之后,不管这个对象原来的构造函数是什么,在深拷贝之后都会变成Object

这种方法能正确处理的对象只有Number, String, Boolean, Array, 扁平对象,即那些能够被json 直接表示的数据结构。RegExp对象是无法通过这种方式深拷贝。

也就是说,只有可以转成JSON格式的对象才可以这样用,像functionundefinedsymbol、循环引用的对象没办法转成JSON

var obj1 = { fun: function(){ console.log(123) } };var obj2 = JSON.parse(JSON.stringify(obj1));console.log(typeof obj1.fun);// 'function'console.log(typeof obj2.fun);// 'undefined' <-- 没复制复制代码
使用递归函数实现一个深拷贝的方法:
function deepClone(obj) {    let objClone = Array.isArray(obj) ? [] : {};    if(obj && typeof obj === "object") {        for(key in obj) {            if(obj.hasOwnProperty(key)) {                if(obj[key] && typeof obj[key] === "object") {                    objClone[key] = deepClone(obj[key]);                } else {                    objClone[key] = obj[key];                    }             }         }     }    return objClone} 复制代码

转载地址:http://zecoo.baihongyu.com/

你可能感兴趣的文章
win10常用快捷键
查看>>
vmware搭建vSAN提示磁盘不合格或者看不到磁盘的解决办法
查看>>
HashMap和Hashtable的区别
查看>>
Oracle EBS-SQL (INV-5):检查期间拉式物料领用记录数.sql
查看>>
Python之with语句原理
查看>>
在Window环境下多线程与CPU资源分配原则
查看>>
20170303新的开始
查看>>
Python--day25--复习(单继承和多继承的总结)
查看>>
@Html.EditFor()不能添加“只读”html属性;以及disable属性的坑
查看>>
Logger日志级别说明及设置方法、说明
查看>>
7-1 列出连通集 (25 分)
查看>>
Mybatis之Mapper动态代理
查看>>
【转】楼天城楼教主的acm心路历程(作为励志用)
查看>>
vw、vh、vmin、vmax 的含义
查看>>
04.设计模式_抽象工厂模式
查看>>
vue项目搭建
查看>>
c lang codesnippets
查看>>
Machine Learning
查看>>
Ext概述
查看>>
LeetCode – Refresh – Populating Next Right Pointers in Each Node I and II
查看>>