Javascript深拷贝

javascript深拷贝是初学者甚至有经验的开发者,都会经常遇到问题,并不能很好的理解javascript的深拷贝。

让客户满意是我们工作的目标,不断超越客户的期望值来自于我们对这个行业的热爱。我们立志把好的技术通过有效、简单的方式提供给客户,将通过不懈努力成为客户在信息化领域值得信任、有价值的长期合作伙伴,公司提供的服务项目有:域名与空间、虚拟空间、营销软件、网站建设、梨林网站维护、网站推广。

深拷贝(deepClone)?

与深拷贝相对的就是浅拷贝,很多初学者在接触这个感念的时候,是很懵逼的。

为啥要用深拷贝?

在很多情况下,我们都需要给变量赋值,给内存地址赋予一个值,但是在赋值引用值类型的时候,只是共享一个内存区域,导致赋值的时候,还跟之前的值保持一直性。

看一个具体的例子

 
 
 
 
  1. // 给test赋值了一个对象
  2. var test = {
  3. a: 'a',
  4. b: 'b'
  5. };
  6.  
  7. // 将test赋值给test2
  8. // 此时test和test2是共享了同一块内存对象,这也就是浅拷贝
  9. var test2 = test;
  10.  
  11. test2.a = 'a2';
  12.  
  13. test.a === 'a2'// 为true 

图解:

这下就很好理解为什么引用值类型数据相互影响问题。

实现

实现一个深拷贝函数,就不得不说javascript的数值类型。

判断javascript类型

javascript中有以下基本类型

类型 描述
undefined undefined类型只有一个值undefined,它是变量未被赋值时的值
null null类型也只有一个值null, 它是一个空的对象引用
Boolean Boolean有两种取值true和false
String 它表示文本信息
Number 它表示数字信息
Object 它是一系列属性的无序集合, 包括函数Function和数组Array

使用typeof是无法判断function和array的,这里使用Object.prototype.toString方法。 默认情况下,每个对象都会从Object上继承到toString()方法,如果这个方法没有被这个对象自身或者更接近的上层原型上的同名方法覆盖(遮蔽),则调用该对象的toString()方法时会返回”[object type]”,这里的字符串type表示了一个对象类型

 
 
 
 
  1. function type(obj) {
  2. var toString = Object.prototype.toString;
  3. var map = {
  4.     '[object Boolean]'  : 'boolean',
  5.     '[object Number]'   : 'number',
  6.     '[object String]'   : 'string',
  7.     '[object Function]' : 'function',
  8.     '[object Array]'    : 'array',
  9.     '[object Date]'     : 'date',
  10.     '[object RegExp]'   : 'regExp',
  11.     '[object Undefined]': 'undefined',
  12.     '[object Null]'     : 'null',
  13.     '[object Object]'   : 'object'
  14. };
  15. return map[toString.call(obj)];

实现deepClone

对于非引用值类型的数值,直接赋值,而对于引用值类型(object)还需要再次遍历,递归赋值。

 
 
 
 
  1. function deepClone(data) {
  2. var t = type(data), o, i, ni;
  3. if(t === 'array') {
  4.     o = [];
  5. }else if( t === 'object') {
  6.     o = {};
  7. }else {
  8.     return data;
  9. }
  10. if(t === 'array') {
  11.     for (i = 0, ni = data.length; i < ni; i++) {
  12.         o.push(deepClone(data[i]));
  13.     }
  14.     return o;
  15. }else if( t === 'object') {
  16.     for( i in data) {
  17.         o[i] = deepClone(data[i]);
  18.     }
  19.     return o;
  20. }

这里有个点大家要注意下,对于function类型,博主这里是直接赋值的,还是共享一个内存值。这是因为函数更多的是完成某些功能,有个输入值和返回值,而且对于上层业务而言更多的是完成业务功能,并不需要真正将函数深拷贝。

但是function类型要怎么拷贝呢?

其实博主只想到了用new来操作一下,但是function就会执行一遍,不敢想象会有什么执行结果哦!o(╯□╰)o!其它暂时还没有什么好的想法,欢迎大家指导哦!

到这里差不多也就实现完了深拷贝,又有人觉的怎么没有实现浅拷贝呢?

浅拷贝?

对于浅拷贝而言,可以理解为只操作一个共同的内存区域!这里会存在危险!(。﹏。*) 。

如果直接操作这个共享的数据,不做控制的话,会经常出现数据异常,被其它部分更改。所以应该不要直接操作数据源,给数据源封装一些方法,来对数据来进行CURD操作。

到这里估计就差不多了,但是作为一个前端,不仅仅考虑javascript本身,还得考虑到dom、浏览器等。

Element类型

来看下面代码,结果会返回啥呢?

 
 
 
 
  1. Object.prototype.toString.call(document.getElementsByTagName('div')[0])

答案是[object HTMLDivElement]

有时候保存了dom元素, 一不小心进行深拷贝,上面的深拷贝函数就缺少了对Element元素的判断。而判断Element元素要使用instanceof来判断。因为对于不同的标签,tostring会返回对应不同的标签的构造函数。

 
 
 
 
  1. function type(obj) {
  2. var toString = Object.prototype.toString;
  3. var map = {
  4.     '[object Boolean]'  : 'boolean',
  5.     '[object Number]'   : 'number',
  6.     '[object String]'   : 'string',
  7.     '[object Function]' : 'function',
  8.     '[object Array]'    : 'array',
  9.     '[object Date]'     : 'date',
  10.     '[object RegExp]'   : 'regExp',
  11.     '[object Undefined]': 'undefined',
  12.     '[object Null]'     : 'null',
  13.     '[object Object]'   : 'object'
  14. };
  15. if(obj instanceof Element) {
  16.         return 'element';
  17. }
  18. return map[toString.call(obj)];

其它方式?

1. jquery的实现

详见https://github.com/jquery/jquery/blob/master/src/core.js

2. underscore的实现

详见https://github.com/jashkenas/underscore/blob/master/underscore.js

3. lodash的实现

详见https://github.com/lodash/lodash/blob/master/lodash.js

4. JSON实现

先通过JSON.stringify一下,然后再JSON.parse一下,就能实现深拷贝。但是数据类型只支持基本数值类型。

 
 
 
 
  1. var obj = {
  2.     a: 'a',    
  3.     b: function(){console.log('b')}
  4. }
  5.  
  6. //在JSON.stringify的时候就会把function给过滤了。
  7.  
  8. JSON.stringify(obj)// "{"a":"a"}" 

小结

这里大概总结了一下深拷贝,以及怎么实现一个深拷贝。在不同的场景下,要根据业务场景,判断是否需要使用深拷贝。

参考文献

winter-JavaScript中的类型 http://www.cnblogs.com/winter-cn/archive/2009/12/07/1618281.html

分享题目:Javascript深拷贝
标题链接:http://www.mswzjz.cn/qtweb/news48/486098.html

攀枝花网站建设、攀枝花网站运维推广公司-贝锐智能,是专注品牌与效果的网络营销公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 贝锐智能