一、曾经在读JQ源码的时候,对深拷贝算是有了一点的理解。我们在项目中是不是经常会遇到这样的问题呢?

后台返回一个数组对象(引用类型).次数在页面渲染中需要对部分数据进行处理 比如:银行卡62345092534 (这么长) 但在页面显示的时候,

只显示中国银行(3118)但是传给后台的时候。又要传623445242整个号码,我们也许会把var oldData = res.data;

但是我们发现两个数据都变了? 这是为什么呢? 其实就是一个深浅拷贝的问题。

二、浅拷贝

比如数组,对象,这样的引用类型。

var arr = ['js','html'];
var oldArr = arr;
oldArr[0] = 'css';
console.log(arr); // ['css','html']
/*
这其实是一个地址的引用。 相当于他们赋值的是指针。
*/
//在数组的方式中可以有两种方式来避免这种问题
//1.slice
var b = ['vue','react'];
var oldB = b.slice(0);
b[0] = 'angular';
console.log(b); //['angular','react']
console.log(oldB); // ['vue','react']
//2.concat();
var c = ['d3','three','webgl'];
var oldC = c.concat([]);
c[0] = 'earthgL';
console.log(c,oldC); // "earthgL", "three", "webgl"] ["d3", "three", "webgl"]

对象的浅复制也是一个道理。是对地址的引用而已。

 var obj = {
name:"前端",
age:"10"
}
var oldObj = obj;
obj.name = "html + css + js";
console.log(oldObj); //{name: "html + css + js", age: "10"}

三、深拷贝。

1.简单来说 深拷贝就事创建了一个新的内存空间。 他们不在会公用同一个内存空间。是两个完全独立的对象或数组。

var defaults = {
name: 'quber',
age: [1, 2, 3, 4],
child: [
{ name: 'qubernet', fun: function () { return 1; } },
{ name: 'qubernet1', fun: function () { return 2; } }
]
};
var newDefaults = $.extend(true, {},defaults );
console.log(JSON.stringify(newDefaults ));

2.JQ深拷贝源码部分

jQuery.extend = jQuery.fn.extend = function() {

    var options, name, src, copy, copyIsArray, clone,
target = arguments[0] || {},
i = 1,
length = arguments.length,
deep = false;
/*
变量 options:指向某个源对象。
变量 name:表示某个源对象的某个属性名。
变量 src:表示目标对象的某个属性的原始值。
变量 copy:表示某个源对象的某个属性的值。
变量 copyIsArray:指示变量 copy 是否是数组。
变量 clone:表示深度复制时原始值的修正值。
变量 target:指向目标对象,申明时先临时用第一个参数值。
变量 i:表示源对象的起始下标,申明时先临时用第二个参数值。
变量 length:表示参数的个数,用于修正变量 target。
变量 deep:指示是否执行深度复制,默认为 false。 ps:源对象指的是把自己的值付给别人的对象;目标对象指的是被源对象赋值的对象
*/ // 如果第一个参数传入的是布尔值
if ( typeof target === "boolean" ) {
deep = target;//设置deep变量,确定是深拷贝还是浅拷贝
target = arguments[1] || {};//将目标对象设为第二个参数值。
i = 2;//源对象的起始下标设为2(即从第三个参数开始算源对象)
} // Handle case when target is a string or something (possible in deep copy)
//嗯,原英文解释的很清楚
if ( typeof target !== "object" && !jQuery.isFunction(target) ) {
target = {};
} // 如果没有目标对象,那么目标对象就是jquery对象
if ( length === i ) {
target = this;
--i;
} 拷贝的核心部分代码
for ( ; i < length; i++ ) {//遍历源对象
// Only deal with non-null/undefined values
if ( (options = arguments[ i ]) != null ) {//options就是源对象
// Extend the base object
for ( name in options ) {//遍历源对象的属性名
src = target[ name ];//获取目标对象上,属性名对应的属性
copy = options[ name ];//获取源对象上,属性名对应的属性 // 如果复制值copy 与目标对象target相等,
//为了避免深度遍历时死循环,因此不会覆盖目标对象的同名属性。
if ( target === copy ) {
continue;
} //递归地将源对象上的属性值合并到目标对象上
//如果是深拷贝,且待拷贝的对象存在,且是普通对象或是数组
//这一个判断条件非常关键,这正是之前疑问的症结
//首先,普通对象的定义是:通过 "{}" 或者 "new Object" 创建的
//回到之前的疑问,目标对象tobeCloned的属性o对象的obj不是普通对象,也不是数组,所以程序不会走到下面的分支
if ( deep && copy && ( jQuery.isPlainObject(copy) || (copyIsArray = jQuery.isArray(copy)) ) ) {
if ( copyIsArray ) {
//如果是数组
copyIsArray = false;
clone = src && jQuery.isArray(src) ? src : []; } else {
clone = src && jQuery.isPlainObject(src) ? src : {};
} // 递归地拷贝
target[ name ] = jQuery.extend( deep, clone, copy ); } else if ( copy !== undefined ) {
//会走到这个分支,这个分支的处理很暴力,就是把源对象的属性,直接赋给源对象。
//对于上文中tobeCloned对象的属性o,没有进一步递归地拷贝,而是直接把引用赋给源对象
//所以改变tobeCloned的o属性时,目标对象的o属性也被改变了。
target[ name ] = copy;
}
}
}
} // Return the modified object
return target;
};

3.SE6提供的深拷贝的方法Object.assign(); (前提是没有属性对象)

四、总结。

简单来说:浅拷贝就是多个变量共用一个地址,深拷贝就是创建了多个内存空间。 (希望对你有帮助)

js 中引用类型 的深拷贝 和 浅拷贝的区别的更多相关文章

  1. 探究JS中对象的深拷贝和浅拷贝

    深拷贝和浅拷贝的区别 在讲深拷贝和浅拷贝的区别之前,回想一下我们平时拷贝一个对象时是怎么操作的?是不是像这样? var testObj1 = {a: 1, b:2}, testObj2=testObj ...

  2. 在js中如何区分深拷贝与浅拷贝?

    一.自我理解 简单来讲就是:深拷贝层层拷贝,浅拷贝只拷贝第一层. 在深拷贝中,新对象中的更改不会影响原对象,而在浅拷贝中,新对象中的更改,原对象中也会跟着改. 在深拷贝中,原对象与新对象不共享相同的属 ...

  3. Python中复制、深拷贝和浅拷贝的区别

    深拷贝定义(deepcopy) 在Python中,由于一切皆对象,所以任何变量都可以被引用,也即可以被赋值给任何变量.但是在Python中,给变量赋值,是区分的,一般情况下,Python中的变量赋值都 ...

  4. C#中??和?分别是什么意思? 在ASP.NET开发中一些单词的标准缩写 C#SESSION丢失问题的解决办法 在C#中INTERFACE与ABSTRACT CLASS的区别 SQL命令语句小技巧 JQUERY判断CHECKBOX是否选中三种方法 JS中!=、==、!==、===的用法和区别 在对象比较中,对象相等和对象一致分别指的是什么?

    C#中??和?分别是什么意思? 在C#中??和?分别是什么意思? 1. 可空类型修饰符(?):引用类型可以使用空引用表示一个不存在的值,而值类型通常不能表示为空.例如:string str=null; ...

  5. 【python测试开发栈】—理解python深拷贝与浅拷贝的区别

    内存的浅拷贝和深拷贝是面试时经常被问到的问题,如果不能理解其本质原理,有可能会答非所问,给面试官留下不好的印象.另外,理解浅拷贝和深拷贝的原理,还可以帮助我们理解Python内存机制.这篇文章将会通过 ...

  6. Python 深拷贝和浅拷贝的区别

    python的复制,深拷贝和浅拷贝的区别    在python中,对象赋值实际上是对象的引用.当创建一个对象,然后把它赋给另一个变量的时候,python并没有拷贝这个对象,而只是拷贝了这个对象的引用  ...

  7. 关于js中for in和foreach in的区别

    js 中for in 和foreach in的区别 两个的作用都用来遍历对象,但为什么有了for in语句了还要foreach in语句呢,后来看了下foreach in开发的文档,foreach i ...

  8. js中加“var”和不加“var”的区别

    JavaScript 拥有动态类型.这意味着相同的变量可用作不同的类型: var x // x 为 undefined var x = 6; // x 为数字 var x = "Bill&q ...

  9. JS中构造函数和普通函数有什么区别

    JS中构造函数有普通函数有什么区别? 1.一般规则 构造函数都应该以 一个大写字母开头,eg: function Person(){...} 而非构造函数则应该以一个小写字母开头,eg: functi ...

随机推荐

  1. /etc/hosts,GoldenGate

    [oracle@g]$ netstat -alp|grep 7809(Not all processes could be identified, non-owned process info wil ...

  2. JAVA框架 json数据交互

    一.导入依赖: <dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId& ...

  3. Vue表单绑定(单选按钮,选择框(单选时,多选时,用 v-for 渲染的动态选项)

    <!DOCTYPE html><html>    <head>        <meta charset="utf-8">      ...

  4. C语言程序设计I—第十三周教学

    第十二周教学总结(26/11-01/12) 第十三周的教学总结在朋友圈发布了,没有及时在此更新,为了保持教学总结的完整性,现补齐. 今日学院有重大外事活动,所有老师停课参加并且不需要补课,但为了保证我 ...

  5. 缩点tarjan

    给定一个n个点m条边有向图,每个点有一个权值,求一条路径,使路径经过的点权值之和最大.你只需要求出这个权值和.允许多次经过一条边或者一个点,但是,重复经过的点,权值只计算一次. 缩点含义:将一个环缩成 ...

  6. day39

    今日内容: 1.对于表,库,记录的基本操作 2.数据库引擎的了解 3.表的详细 4.数据类型的掌握 1.回顾昨日对于表,库,记录的基本操作 库 增: create database mydb2; 删: ...

  7. js 动态声明变量(eval)

    eval()可以将一段字符串当作js代码来执行. 动态声明变量(字符串)可通过eval(字符串)来实现.举例如下: var Thread_num=5; for(var i=1;i<=Thread ...

  8. kettle学习笔记(九)——子转换、集群与变量

    一.概述 kettle中3个重要的步骤: 子转换/映射 在转换里调用一个子转换,便于封装和重用. 集群 集群模式 变量和参数 变量和参数的用法 二.子转换 1.定义子转换 主要由映射输入与映射输出定义 ...

  9. Android应用安全之第三方SDK安全

    第三方sdk的包括广告.支付.统计.社交.推送,地图等类别,是广告商.支付公司.社交.推送平台,地图服务商等第三方服务公司为了便于应用开发人员使用其提供的服务而开发的工具包,封装了一些复杂的逻辑实现以 ...

  10. 20155234 昝昕明《基于ARM实验箱的国密算法应用》课程设计个人报告

    20155234 昝昕明<基于ARM实验箱的国密算法应用>课程设计个人报告 个人贡献 参与课设题目讨论及完成全过程: 资料收集: SM1算法及和ARM之间通信 负责串口代码调试: 协调完成 ...