要用递归实现深拷贝,首先说说什么是深拷贝和浅拷贝

浅拷贝:一个值赋给另一个值,当原先的值不改变地址的情况下改变数据,另一个值跟着变
深拷贝:一个值赋给另一个值,当原先的值不改变地址的情况下改变数据,另一个值不变
 
注意 : 强调一点,原先的值不改变地址的情况下改变数据,如果是改了地址的改变,那将用另一个值的改变与否判断深拷贝浅拷贝毫无意义
 
// 举一个简单的例子---以下的obj,obj2可以看做是这里的obj,obj2

var obj = {
name : 'zs',
age : 18
}
// 浅拷贝
var obj2 = obj;
obj.name = 'ls';
console.log(obj2) // { name : 'ls' , age : 18 }

这是浅拷贝,单单就是把obj的地址给了obj2,obj改了,obj2也要改。再来说说这obj改了,这里面有点不同。要么 obj改了属性(数组就是改了元素),要么干脆obj指向另一个地址,例如:obj = 1;针对第一种就是深浅拷贝可以作为判断依据的,根据改了之后obj2改不改就可以看出深拷贝(不改)还是浅拷贝(改了)。第二种不管深浅拷贝的因为obj的地址变了,obj再怎么改都不会影响obj2,因此obj2肯定不会变,深浅拷贝看不出来的

var obj = {
name : 'zs',
age : 18
}
var obj2 = obj;
// obj 自己地址都变了,再怎么折腾影响不了obj2了
obj = 1;
console.log(obj2) // { name : 'zs' , age : 18 }

//  以上 obj可以看做第一个对象(数组) ,obj2 可以看做第二个对象(数组) ,为什么不说函数呢? 因为函数本身办法改内容,声明的时候已经确定好函数体了,你还能有神通在不改变声明函数的函数体前提下,后续添加函数体语句?如果可以,请教教我,说真的!

----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

以下重点说说深拷贝 : 说白了深拷贝的目的就是不想让obj来影响obj2,有点过河拆桥的意思,把好处或爱情给了另一个人,另一个人却不想让你打扰她(说多了是泪)

针对以下三种常见情况提供三种方式进行深拷贝 : 

很纯的对象 : 属性值没有数组、对象的   var obj2 = Object.assgin({ }, obj)

没有方法的对象 (没有方法在内,可以有数组和对象充当属性值) : var obj2 = JSON.parse(JSON.stringify(obj))

        var u = {
o : {'name' : 'zs'},
i : 5
}
var v = JSON.parse(JSON.stringify(u))
u.o.name = 'ls';
console.log(u,v); // v的o属性中的name还是'zs'

③ 递归( 通用,从底层的角度 ) 基本思想 : 判断obj 是基本类型就简单赋值,对象(数组)的话 ,创建一个空[ ](obj是对象)或空{ }(obj是对象),再判断obj的属性有没有对象(数组),有递归调用,没有简单赋值,直到全部赋值完毕,返回赋值后的new_obj , 看不懂没关系下面代码有注释,绝对可以看懂

var obj = {
arr1 : [1,2,3],
fn: function(){
console.log('我是一个方法')
},
a : '我是普通属性'
} // 现在我要把obj字面量创建里的属性深拷贝( 属性值是引用类型也要深拷贝 )
function deepClone(obj){
// 根据类型制造一个新的数组或对象 => 指向一个新的空间
// 由于数组的typeof也是'object',所以用Array.isArray(obj)
var new_obj = Array.isArray(obj) ? [] : {};
// 首先判断obj的类型
// 普通类型
if( typeof obj != 'object' ){
// 这里不能直接返回obj,不然就是浅拷贝的性质
return new_obj = obj
}
//引用类型
//数组
if(obj instanceof Array ){
for(i = 0; i < obj.length; i++ ){
new_obj[i] = obj[i];
if(typeof new_obj[i] == 'object'){
deepClone(new_obj[i])
}
}
}else{ //对象
for (let key in obj) {
if (obj.hasOwnProperty(key)) {
// 对象中的数组和对象
if (typeof obj[key] == 'object') {
new_obj[key] = deepClone(obj[key]);
}else{//对象中没有引用类型
new_obj[key] = obj[key]
}
}
}
}
return new_obj;
}
var deepClone = deepClone(obj);
console.log(deepClone);
// 测试是不是深拷贝
obj.fn = '我改变了方法属性';
console.log(obj); //{arr1: Array(3), fn: ƒ, a: "我是普通属性", c: "我新增了一个属性"}
console.log(deepClone); // 还是 {arr1: Array(3), fn: ƒ, a: "我是普通属性"}

-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

本文原创,仅供学习使用

递归实现深拷贝( 只要学过js递归,看不懂找我包会 )的更多相关文章

  1. js递归遍历多维数组并在修改数组的key后返回新的多维数组

    我司最近正在用VUE做一个基于用户权限显示不同左侧菜单的后台管理系统,接口会根据用户的权限不同返回不同的菜单名称.URL等,前端要将这些菜单名称及URL动态添加到系统的左侧,这里就用到了vue-rou ...

  2. 【Vue.js实战案例】- Vue.js递归组件实现组织架构树和选人功能

    大家好!先上图看看本次案例的整体效果. 浪奔,浪流,万里涛涛江水永不休.如果在jq时代来实这个功能简直有些噩梦了,但是自从前端思想发展到现在的以MVVM为主流的大背景下,来实现一个这样繁杂的功能简直不 ...

  3. js 递归调用

    js递归调用 function fact(num) { ) { ; } else { ); } } 以下代码可导致出错: var anotherFact = fact; fact = null; al ...

  4. 关于js回调方法 js递归时使用方法

    js中递归调用本身可以这样: function a1(n){ a1(n)}但是如果需要在参数n进行自增的情况下判断会出错: function a1(n){ if(n>10) return 'aa ...

  5. javascript之闭包,递归,深拷贝

    闭包 理解:a函数执行后return出b函数且b函数可以访问a函数的数据 好处:子函数存储在复函数内部,子函数执行完不会被自动销毁 坏处:占用内存比较大 ex: function bibao(){ v ...

  6. JS递归及二叉搜索树的移除节点

    1递归含义:在某时某刻某个条件下调用包含自己的函数 2:注意点:⑴递归过程中一定要加限制条件,要不然会陷入死循环: 死循环eg: function f(someP){ f(somP); } f(4); ...

  7. SQL Server中公用表表达式 CTE 递归的生成帮助数据,以及递归的典型应用

    本文出处:http://www.cnblogs.com/wy123/p/5960825.html 我们在做开发的时候,有时候会需要一些帮助数据,必须需要连续的数字,连续间隔的时间点,连续的季度日期等等 ...

  8. 【高德地图API】从零开始学高德JS API(七)——定位方式大揭秘

    原文:[高德地图API]从零开始学高德JS API(七)——定位方式大揭秘 摘要:关于定位,分为GPS定位和网络定位2种.GPS定位,精度较高,可达到10米,但室内不可用,且超级费电.网络定位,分为w ...

  9. 【高德地图API】从零开始学高德JS API(八)——地址解析与逆地址解析

    原文:[高德地图API]从零开始学高德JS API(八)——地址解析与逆地址解析 摘要:无论是百度LBS开放平台,还是高德LBS开放平台,其调用量最高的接口,必然是定位,其次就是地址解析了,又称为地理 ...

随机推荐

  1. Contentprovider 注册 启动简单流程

    安装app时packagemanager 读取manixfest获取provider信息 存在数据库里流程:1.加载ActivityThread main方法,创建消息队列.ActivityThrea ...

  2. yum源配置及详解

      红帽系列中,进行软件安装可以有三种方法,编译安装,rpm包安装,和yum源安装.其中yum方法安装最简单,因为它可以自动解决软件包之间的依赖关系... 一.常用yum源 yum源可以来源于多种文件 ...

  3. vue2-vux-fitness-project

    非常感谢那些无私开源的程序员,希望我也能够有能力像你们那样,开源很多很有意思的东西~~ //index.html <!DOCTYPE html> <html> <head ...

  4. 深入理解spring注解之@ComponentScan注解

    今天主要从以下几个方面来介绍一下@ComponentScan注解: @ComponentScan注解是什么 @ComponentScan注解的详细使用 1,@ComponentScan注解是什么 其实 ...

  5. Apple的App Analytics统计平台你必须知道的

    Apple最近在iTunesConnect里最新发布了App Analytics统计平台,提供了现有友盟统计平台和自有统计平台无法统计的数据,具有自己的独有特点,尤其是下面几个最让人头疼的流量分析转化 ...

  6. 2019-5-25-win10-uwp-win2d-入门-看这一篇就够了

    title author date CreateTime categories win10 uwp win2d 入门 看这一篇就够了 lindexi 2019-5-25 20:0:52 +0800 2 ...

  7. 数据人看Feed流-架构实践

    背景 Feed流:可以理解为信息流,解决的是信息生产者与信息消费者之间的信息传递问题.我们常见的Feed流场景有:1 手淘,微淘提供给消费者的首页商品信息,用户关注店铺的新消息等2 微信朋友圈,及时获 ...

  8. SGU 107 987654321 problem【找规律】

    题目链接: http://acm.sgu.ru/problem.php?contest=0&problem=107 题意: 平方后几位为987654321的n位数有多少个 分析: 虽然说是水题 ...

  9. ROS报错:IOError:[Errno 13]permission denied: /home/neousys/.ros/roscore-11311.pid"

    在安装ROS后启动ROS,输入:roscore 时报错: 这个问题是由于该路径下ros文件权限造成的. 输入以下命令修改权限: sudo chmod -R ~/.ros/ 修改完成后再次输入rosco ...

  10. @loj - 2093@ 「ZJOI2016」线段树

    目录 @description@ @solution@ @accepted code@ @details@ @description@ 小 Yuuka 遇到了一个题目:有一个序列 a1,a2,..., ...