JavaScript浅层克隆与深度克隆示例详解


Posted in Javascript onSeptember 01, 2020

1 相关知识点

  1. 浅克隆就是将栈内存中的引用复制一份,赋给一个新的变量,本质上两个指向堆内存中的同一地址,内容也相同,其中一个变化另一个内容也会变化。
  2. 深克隆就是创建一个新的空对象,开辟一块内存,然后将原对象中的数据全部复制过去,完全切断两个对象间的联系。
  3. 区别:浅克隆和深克隆最大的区别就是对引用值的处理了,即浅克隆之后你改我也改,深克隆之后你改我不改。(PS:原始值的处理一样)
  4. 原始值(栈数据stack):Number,Boolean(false/true),String,undefined,null
  5. 引用值(堆数据heap):Array,Object,function ··· Date,RegExp

2 浅层克隆

在浅层克隆中,原始值的克隆没问题,只是值的拷贝,不会出现你改我改的问题。但是引用值的克隆,就会出现你改我也改的问题,因为浅层克隆的是地址,即指向的是同一空间。

2.1 浅克隆函数

function clone(origin, target) {
  var target = target || {}; 
  //容错,即防止用户不传递目标参数。若用户传递了参数就用,若没传则拿一个空对象当目标
  for (var prop in origin) {
    target[prop] = origin[prop];
  }
  return target;
}

2.2 运用实例

function clone(origin, target) {
  var target = target || {}; 
  for (var prop in origin) {
    target[prop] = origin[prop];
  }
  return target;
}

var obj = {
  name: 'abc',
  age: '18',
  sex: 'male',
  card: ['a', 'b', 'c'],
  book: {
    name: 'ccc',
    sbook: {
      name: 'aaa'
    }
  }
};
var newobj = {};

clone(obj, newobj);

运行代码如下:

JavaScript浅层克隆与深度克隆示例详解

3 深度克隆

进行深度克隆之后,对于引用值的克隆的问题就会和原始值一样我改你不改,因为在深度克隆中虽然是相同的东西,但是指向不同的空间。即深度克隆之后,值各自独立,互不影响。

3.1 深克隆步骤分析

需要进行深度克隆的对象如下:

var obj = {
  name: 'abc', // 原始值
  age: '18', // 原始值
  sex: 'male',// 原始值
  card: ['a', 'b', 'c'], // 引用值
  book: { // 引用值
    name: 'ccc', // 原始值
    sbook: { // 引用值
      name: 'aaa'// 原始值
    }
  }
};

var obj1 = {};

(1)首先需要遍历要克隆的对象

方法:for (var prop in origin){···}

for (var prop in origin) {
	···
}

(2)依次判断是不是原始值

方法:typeof() ,即若为原始值,就直接拷贝;若为引用值(typeof(···)返回的值是object),则进行递归操作。需要注意是的typeof(null) == 'object',所以得排除这一个情况。

if (origin[prop] !== "null" && typeof(origin[prop]) == 'object') {
	···
	// 递归
} else {
  target[prop] = origin[prop];
}

(3)判断是数组还是对象

方法:toString(推荐), constructor,instanceof (后两个会涉及到父子域的小问题,虽然遇到的可能不是很大)

var toStr = Object.prototype.toString,
  arrStr = "[object Array]";
if (toStr.call(origin[prop]) == arrStr) {
	··· // 数组
} else {
	··· // 对象
}

(4)建立相应的数组或对象

方法:建立一个新的同名空数组 / 对象,并将原始对象中的 数组或对象 当成一个新的原始对象,再次将其中的数据拷贝到目标对象的 同名空数组 / 对象 里面。即递归开始拷贝数组或对象里面的数据,并递归执行第(1)步。递归完成之后,再依次进行下一个数据的克隆。

var toStr = Object.prototype.toString,
  arrStr = "[object Array]";
if (toStr.call(origin[prop]) == arrStr) {
	target[prop] = [];
} else {
	target[prop] = {};
}
newobj = {
  name: 'abc',
  age: '18',
  sex: 'male',
  card: [] 
  // 建立一个新的同名空数组,并把obj的card数据当成一个原始对象,再次拷贝到obj1的card里面
  // 即 递归开始拷贝数组或对象里面的数据,递归执行第(1)步
  // 执行完数组card拷贝之后,开始同理拷贝下一个对象book···
}

3.2 深克隆函数

function deepClone(origin, target) {
  var target = target || {},
    toStr = Object.prototype.toString,
    arrStr = "[object Array]";
  for (var prop in origin) {
    if (origin.hasOwnProperty(prop)) {
      if (origin[prop] !== "null" && typeof(origin[prop]) == 'object') {
        if (toStr.call(origin[prop]) == arrStr) {
          target[prop] = [];
        } else {
          target[prop] = {};
        }
        deepClone(origin[prop], target[prop]);

      } else {
        target[prop] = origin[prop];
      }
    }
  }
  return target;
}

使用三目运算符简化后的代码如下:

// 使用三目运算符简化后
function deepClone(origin, target) {
  var target = (target || {}),
    toStr = Object.prototype.toString,
    arrStr = "[object Array]";
  for (var prop in origin) {
    if (origin.hasOwnProperty(prop)) {
      if (origin[prop] !== "null" && typeof (origin[prop]) == 'object') {
        target[prop] = toStr.call(origin[prop]) == arrStr ? [] : {};
        deepClone(origin[prop], target[prop]);
      } else {
        target[prop] = origin[prop];
      }
    }
  }
  return target;
}

3.3 运用实例

// 使用三目运算符简化后
function deepClone(origin, target) {
  var target = (target || {}),
    toStr = Object.prototype.toString,
    arrStr = "[object Array]";
  for (var prop in origin) {
    if (origin.hasOwnProperty(prop)) {
      if (origin[prop] !== "null" && typeof (origin[prop]) == 'object') {
        target[prop] = toStr.call(origin[prop]) == arrStr ? [] : {};
        deepClone(origin[prop], target[prop]);
      } else {
        target[prop] = origin[prop];
      }
    }
  }
  return target;
}

运行代码如下:

JavaScript浅层克隆与深度克隆示例详解

3.4 hasOwnProperty

hasOwnProperty() 方法会返回一个布尔值,指示对象自身属性中是否具有指定的属性(即是否有指定的键)。

语法:obj.hasOwnProperty(prop)
参数:要检测的属性的字符串形式表示的名称,或者Symbol。
返回值:用来判断某个对象是否含有指定的属性的布尔值。
描述:所有继承了Object的对象都会继承到hasOwnProperty方法。这个方法可以用来检测一个对象是否含有特定的自身属性;和in运算符不同,该方法会忽略掉那些从原型链上继承到的属性。
用法:
a. 使用hasOwnProperty方法判断属性是否存在
b. 区别自身属性与继承属性
c. 遍历一个对象的所有自身属性
d. 使用hasOwnProperty作为属性名
具体知识点请参考 Object.prototype.hasOwnProperty()

若对象里面编写了原型属性,但遍历的时候并不想让其显示出来,就可以使用对象名.hasOwnProperty(属性名) 来判断是否是自身属性,若是自己的则返回值为true,若不是自身原型属性则返回值为false。实例如下:

var obj = {
	name: 'ABC',
	age: '18',
	sex: 'male',
	__proto__: {
		heart: 'happy'
	}
}
for (var prop in obj) {
	// 配套使用,起到一个过滤的作用,不把原型链上的数据弄出来
	if (obj.hasOwnProperty(prop)) {
		console.log(obj[prop]);// ABC 18 male
	}
}

个人笔记,欢迎大家交流探讨!

总结

到此这篇关于JavaScript浅层克隆与深度克隆的文章就介绍到这了,更多相关JavaScript浅层克隆与深度克隆内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Javascript 相关文章推荐
js获取变量
Aug 24 Javascript
js 文件引入实现代码
Apr 23 Javascript
jQuery新的事件绑定机制on()示例应用
Jul 18 Javascript
Javascript核心读书有感之表达式和运算符
Feb 11 Javascript
jQuery实现自动滚动到页面顶端的方法
May 22 Javascript
jQuery仅用3行代码实现的显示与隐藏功能完整实例
Oct 08 Javascript
利用jQuery设计一个简单的web音乐播放器的实例分享
Mar 08 Javascript
jQuery滚动新闻实现代码
Jun 26 Javascript
学习Javascript闭包(Closure)知识
Aug 07 Javascript
常用Javascript函数与原型功能收藏(必看篇)
Oct 09 Javascript
JavaScript实现换肤功能
Sep 15 Javascript
移动端滑动切换组件封装 vue-swiper-router实例详解
Nov 25 Javascript
VUE子组件向父组件传值详解(含传多值及添加额外参数场景)
Sep 01 #Javascript
vue离开当前页面触发的函数代码
Sep 01 #Javascript
Vue 实现监听窗口关闭事件,并在窗口关闭前发送请求
Sep 01 #Javascript
Node.js web 应用如何封装到Docker容器中
Sep 01 #Javascript
Vue中关闭弹窗组件时销毁并隐藏操作
Sep 01 #Javascript
Vue 使用typescript如何优雅的调用swagger API
Sep 01 #Javascript
vue路由切换时取消之前的所有请求操作
Sep 01 #Javascript
You might like
php用户注册时常用的检验函数实例总结
2014/12/22 PHP
PHP调用Linux命令权限不足问题解决方法
2015/02/07 PHP
PHP实现C#山寨ArrayList的方法
2015/07/16 PHP
Yii2简单实现多语言配置的方法
2016/07/23 PHP
PHP实现的链式队列结构示例
2017/09/15 PHP
详细解读php的命名空间(一)
2018/02/21 PHP
laravel-admin select框默认选中的方法
2019/10/03 PHP
javascript自执行函数之伪命名空间封装法
2010/12/25 Javascript
浅析基于WEB前端页面的页面内容搜索的实现思路
2014/06/10 Javascript
AngularJS + Node.js + MongoDB开发的基于高德地图位置的通讯录
2015/01/02 Javascript
JQuery实现网页右侧随动广告特效
2016/01/17 Javascript
jQuery图片拖动组件Dropzone用法示例
2017/01/17 Javascript
jQuery is not defined 错误原因与解决方法小结
2017/03/19 Javascript
Angular2利用组件与指令实现图片轮播组件
2017/03/27 Javascript
Angular2使用Angular CLI快速搭建工程(一)
2017/05/21 Javascript
Ajax高级笔记 JavaScript高级程序设计笔记
2017/06/22 Javascript
关于Google发布的JavaScript代码规范你要知道哪些
2018/04/04 Javascript
linux系统使用python监测系统负载脚本分享
2014/01/15 Python
用python + hadoop streaming 分布式编程(一) -- 原理介绍,样例程序与本地调试
2014/07/14 Python
探究数组排序提升Python程序的循环的运行效率的原因
2015/04/01 Python
设计模式中的原型模式在Python程序中的应用示例
2016/03/02 Python
详解分布式任务队列Celery使用说明
2018/11/29 Python
pycharm打开命令行或Terminal的方法
2019/01/16 Python
python中复数的共轭复数知识点总结
2020/12/06 Python
英国发展最快的在线超市之一:Click Marketplace
2021/02/15 全球购物
交通事故协议书
2014/04/15 职场文书
关于保护环境的标语
2014/06/09 职场文书
中国在我心中演讲稿
2014/09/13 职场文书
再婚婚前财产协议书范本
2014/10/19 职场文书
2014年远程教育工作总结
2014/12/09 职场文书
药店营业员岗位职责
2015/04/14 职场文书
居委会工作总结2015
2015/05/18 职场文书
童年读书笔记
2015/06/26 职场文书
干部考核工作总结
2015/08/12 职场文书
远程教育学习心得体会
2016/01/23 职场文书
Win11运行cmd提示“请求的操作需要提升”的两种解决方法
2022/07/07 数码科技