JavaScript之深拷贝&浅拷贝
深拷贝&浅拷贝,说起来都明白,但是说不出所以然。今天就系统的整理下思绪,一点点的将其分析出所以然
废话不多说
浅拷贝
简单的说就是一个值引用,学生时代接触过编程的人都应该了解过指针,浅拷贝可以说就是变量拷贝的是数据的地址而不是数据本身,所以从直观上看来,好像是一个数据改变了, 所有数据都改变了
var obj1 = {
str: "good"
}
var obj2 = obj1;
console.log(obj2.str); //good
obj2.str = "bad";
console.log(obj1.str); //bad
通过浅拷贝 A 和 B 指向相同的内存地址,通过 A (或 B)对数据进行修改,相应的在 B (或 A)可以得到体现
那么就 JavaScript 中来说,浅拷贝一般是指数组对象的浅拷贝,因为基本数据类型的赋值都是值传递(所以深拷贝都是一层一层的拨开对象或数组,直到基本数据类型然后进行赋值拷贝,完成)
function copy(obj) {
var cc = {};
for(var i in obj) {
cc[i] = obj[i];
}
return cc;
}
var obj1 = {
name: "yellow",
arr: ["北京", "上海", "广州"]
}
var obj2 = copy(obj1);
obj2.name = "blue";
obj2.arr.push("深圳");
此时,对象 obj1 和 obj2 的如下图所示, 可以看到属性 name 的值是不一样的,但是属性 arr 由于是一个数组,两个对象里都多了一个“深圳”,因为他们都是一个值引用, 两个对象里的arr 指向同一块内存地址,所以通过其中一个改变其值,另一个的值也会跟着改变。这是一个非常常见的一个浅拷贝的例子。
到这里,我们可以总结一下:浅拷贝,对于基本数据类型是值传递,而对于引用类型(对象等)是进行的引用传递。
深拷贝
深拷贝,从字面看,就是比浅深一点。
深拷贝与浅拷贝从我们直观测试看得到的地方来说,区别就是:深拷贝之后,两个值的变化互不影响
怎样达到这个目的呢?就是在拷贝的时候,是把值传递出去,还不是单纯的指向值的位置,就像上边的 name 值一样。
可以对比下上边两段代码的 str 属性和 name 属性,发现了什么么。在第一段代码中 str 属性是放在对象 obj1 里边进行的赋值,是以对象 obj1 为单位的,第二段代码中,name 属性是以name 属性本身为单位的赋值,name 本身是一个字符串类型(基本数据类型),所以进行的是值传递。
所以, 如果在进行赋值拷贝时,如果一个值是引用类型(对象), 那么就“刨”一层,直到遇到基本数据类型,然后进行赋值。从而使整个拷贝都是基于基本数据类型进行的,最终达到我们深拷贝的结果
上一段简单的深拷贝示例代码
function deepCopy(obj) {
if(typeof obj !== "object" && obj === null) {
return;
} else {
var cc = obj.constructor === Array?[]:{};
}
for(var i in obj) {
if(typeof obj[i] === "object") {
cc[i] = deepCopy(obj[i]);
} else {
cc[i] = obj[i];
}
}
return cc;
}
以上是个人对深拷贝&浅拷贝的理解,欢迎交流点评。
JavaScript之深拷贝&浅拷贝的更多相关文章
- javascript对象深拷贝,浅拷贝 ,支持数组
javascript对象深拷贝,浅拷贝 ,支持数组 经常看到讨论c#深拷贝,浅拷贝的博客,最近js写的比较多, 所以也来玩玩js的对象拷贝. 下面是维基百科对深浅拷贝的解释: 浅拷贝 One meth ...
- JavaScript实现深拷贝(深复制) 面试题
1.两种方法实现深拷贝(深复制) (1)方法一:兼容性好,请仔细看代码(网上大部分代码有Bug) (2)方法二:需要对象满足JSON数据格式.JOSN数据格式:http://www.cnblogs.c ...
- JS Object Deep Copy & 深拷贝 & 浅拷贝
JS Object Deep Copy & 深拷贝 & 浅拷贝 https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Refe ...
- c# 内存的具体表现- 通用类型系统 深拷贝 浅拷贝 函数传参
c# 通用类型系统 及变量在 深拷贝 浅拷贝 函数传参 中的深层次的表现 在编程中遇到了一些想不到的异常,跟踪发现,自己对于c#变量在内存上的表现理解有偏差,系统的学习并通过代码实验梳理了各种情况下, ...
- python集合增删改查,深拷贝浅拷贝
集合 集合是无序的,不重复的数据集合,它里面的元素是可哈希的(不可变类型),但是集合本身是不可哈希(所以集合做不了字典的键)的.以下是集合最重要的两点: 去重,把一个列表变成集合,就自动去重了. 关系 ...
- 【opencv】imread 赋值 深拷贝浅拷贝
import cv2 import copy import os def filter_srcimg(dstimg): ss=3 srcimg=copy.deepcopy(dstimg) #aa=5 ...
- Java基础 深拷贝浅拷贝
Java基础 深拷贝浅拷贝 非基本数据类型 需要new新空间 class Student implements Cloneable{ private int id; private String na ...
- 【04】Python 深拷贝浅拷贝 函数 递归 集合
1 深拷贝浅拷贝 1.1 a==b与a is b的区别 a == b 比较两个对象的内容是否相等(可以是不同内存空间) a is b 比较a与b是否指向同一个内存地址,也就是a与b的id是否相 ...
- JavaScript中深拷贝实现
JavaScript 中深拷贝实现 拷贝时候涉及到: 1.循环结构 2.判断数组 Array 还是对象 Object 函数实现 /** * 获取满足条件的数组中的第一个元素 * @param ...
随机推荐
- vue双向绑定原理及实现
vue双向绑定原理及实现 一.总结 一句话总结:vue中的双向绑定主要是通过发布者-订阅者模式来实现的 发布 订阅 1.单向绑定和双向绑定的区别是什么? model view 更新 单向绑定:mode ...
- Rspec: everyday-rspec实操: 第9章 快速编写测试,编写快速的测试。
Make it work, make it right, make it fast. 测试运行的时间.应用和测试组件的增长,速度会越来越慢,目标是保持代码的readable, maintainable ...
- hdu1564博弈+找规律
#include<map> #include<set> #include<cmath> #include<queue> #include<stac ...
- echarta3 北京,上海地图
1.首先你得到echarts官网下载js,建议下载完整代码,这样你就很容易根据我的路径找到beijing.js 2.把echarts.js和beijingi.js根据你的路径引对,然后就可以copy我 ...
- Uedit个人专注
Uedit个人专注 Windows Registry Editor Version 5.00 [HKEY_CLASSES_ROOT\*\Shell\Uedit] [HKEY_CLASSES_ROO ...
- c# winform 操作oracle数据库的Blob字段,把图片存储到数据库,保存图片到数据库
///c# winform 操作oracle数据库的Blob字段,把图片存储到数据库,保存图片到数据库 闲话不多说,直接上代码 using System; using System.Collectio ...
- Java 动态代理与反射机制
java动态代理必须的两个类与两个接口: 首先需要有一个接口(委托者需要实现该接口的方法)示例如下: <pre name="code" class="html&qu ...
- C#如何弹出输入框
在C#中,进行windows窗体应用程序编程的时候,经常需要弹出输入框,输入密码,输入文本之类的.然而,C#中没有直接弹出输入框的语句,MessageBox只能显示一段消息而不能输入.我们需要调用Mi ...
- BootStrap--scroll
滚动侦测 滚动侦测基本使用方法为: <body data-spy="scroll"> <nav class="navbar navbar-default ...
- win10下安装MySQL5.7.20
1. 下载Mysql官方:http://www.mysql.com→downloads→选社区版本MySQL Community Edition(GPL)→点击Community(GPL)Downlo ...