递归实现深度克隆(内含JS数据类型判断,对象遍历)

转自:http://blog.csdn.net/qq_27090183/article/details/50823429


使用递归来实现一个深度克隆,可以复制一个目标对象,返回一个完整拷贝
被复制的对象类型会被限制为数字、字符串、布尔、日期、数组、Object对象。不会包含函数、正则对象等

首先要去判断要克隆的对象的值类型或者引用类型。判断方法有很多种!

对于值类型或者引用有4种方法判断:

1.typeof

但是!js的数值有两种构造方法:直接赋值法和通过值函数构造器构造
例如:

var test1 = "string"; 
var test2 = new String("string2"); 
console.log(typeof test1);//输出string
console.log(typeof test2);//输出object

对于typeof来说;所有通过构造器constructor产生的变量都是object.那么我们怎么去判断用constructor产生的变量?

2.instanceof

instanceof 函数可以判断左边参数是否是右边参数的一个实例!
例如:

console.log(test2 instanceof String);//输出true
console.log(test1 instanceof String);//输出false

这就很不和谐了!有没有两种都能判断的方法呢?

3.Object.prototype.toString.call()

当然有!MDN在官方教程上就介绍了一种可以判断所有类型的方法!
使用toString()方法来检测对象类型

console.log(Object.prototype.toString.call(test1))//输出[object String] 
console.log(Object.prototype.toString.call(test1))//输出[object String]

顿时觉得人生豁然开朗了起来!啊~五环!你比四环多一环!

4.Object.constructor

在研究MDN 的api文档的时候,发现了constructor方法!
无论怎么样,这个方法都返回一个指向创建了该对象原型的函数引用。

需要注意的是,该属性的值是那个函数本身, 而不是一个包含函数名称的字符串!
对于原始值(如1,true 或 “test”),该属性为只读。 所有对象都会从它的原型上继承一个constructor属性. 所以,虽然可以实现判断,但是还是不用为好!

console.log(test1.constructor.toString()=="function String() { [native code] }")//true; 
console.log(test2.constructor.toString()=="function String() { [native code] }")//true;

总结:方法3、4对于所有值类型和引用类型适用(推荐第三种方法,毕竟官方),第12种方法看情况使用!

然后来到遍历问题了!
1.字符串的遍历

 var temp = src.split(""); 
var cloneString=""; 
for(var i=0;i<temp.length;i++) 
{ 
    cloneString+=temp[i]; 
}

原理就是利用split()方法将字符串里一个个字母分开(注意里面的参数为空 "",为其他就会以这个参数为标准分离字符串)

2.数组的遍历
使用的是传统的数组遍历

var temp = new Array(); 
for(var i=0,a;a = array[i];i++) 
{ 
    temp[i] = cloneObject(a); 
}

这里遇到了几个坑,先说一下: 当使用

for(var a in array)
{
    console.log(a+typeof a)
}

得到的值是 0 string 1 string 2 string 这种方法并不行 还有一个坑就是当使用
for(var i=0,a;a = array[i++];)时 i会在a被赋值后就自动增加而不是等到一个循环完成再增加 ;也就是遍历结果是对的,但是i的数值变化是从1开始而不是从0开始的!
在赋值的过程中,我首先使用的是temp.push()方法!但是!push方法会让temp数组新增加的元素的类型为undefined!这不是我想要的结果!我要的是完美克隆,即数组里面对象类型也要和原来的一致。

看了MDN的api接口发现解决方法如下:
Array.prototype.push.apply(temp,array);

3.对象的遍历
var temp = {}; var keys = Object.keys(src); // keys 为对象src的键名字数组
// 它是数组!!!

for(var i=0,a;a=keys[i];i++) 
{
    temp[a] = cloneObject(src[a]); 
}

对象的遍历,首先获得它的键数组(对象自带的keys()方法),然后再通过键遍历一次值就行了,很简单。

多看MDN总会有收获!下面附上我的代码!

var cloneObject = function(src){
    var Result;
    switch(Object.prototype.toString.call(src)){
        case "[object Number]": 
            Result = (typeof src === "object"?new Number(src):parseInt(src.toString()));
            break;
        case "[object String]":
            // 遍历字符串 =.= 好像没啥意义
            // {
            //  var temp = src.split("");
            //  var cloneString="";
            //  for(var i=0;i<temp.length;i++)
            //  {
            //      cloneString+=temp[i];
            //  }
            // }
            Result = (typeof src === "object"?new String(src):src.toString());
            break;
        case "[object Boolean]":
            Result = (typeof src === "Boolean"?new Boolean(src):src);
            break;
        case "[object Date]":
            Result = new Date(src);
            break;
        case "[object Array]":
            var temp = new Array();
              // Array.prototype.push.apply(temp,src);
             // 当使用for(var i=0,a;a = src[i++];) i会在a被赋值后就自动增加而不是
             // 等到一个循环完成再增加
            for(var i=0,a;a = src[i];i++)
            {
                  // temp.push(cloneObject(a));
                  // 使用push方法会让数组所有元素的类型变成undfined
                 temp[i] = cloneObject(a);
            }
            Result = temp;
            delete temp;
            break;
        case "[object Object]":
            var temp = {}; 
            var keys = Object.keys(src);
            // keys 为对象src的键名字数组
            // 它是数组!!!
            for(var i=0,a;a=keys[i];i++)
            {
                temp[a] = cloneObject(src[a]);
            }
            Result = temp;
            delete temp;
            delete keys;
            break;
        default:
            break;
    }
    return Result;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,905评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,140评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,791评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,483评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,476评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,516评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,905评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,560评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,778评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,557评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,635评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,338评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,925评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,898评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,142评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,818评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,347评论 2 342

推荐阅读更多精彩内容

  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 10,849评论 6 13
  • 1.通过typeof可以判断处几种基本数据类型Boolean,number,string,null,undefin...
    舟渔行舟阅读 623评论 0 1
  • 中山路步行街逛吃逛吃 【1.莲欢海蛎煎】放弃了唾手可得的“黄则和”、“捞海坞”、“黄榕树下”……坚持寻到这家...
    i小森阅读 825评论 3 9
  • 太阳刚刚升到头顶的大中午,一个人坐在空旷的自习室里,捧着书咬着笔看着黑板默默发呆。这样一个懒散的生活状态...
    工科小虾阅读 322评论 25 2
  • 李白的剑 他是我最好的朋友。身高175,不胖不瘦。卷卷的头发,深凹的眼眶,挺翘的鼻梁,薄薄的嘴唇。 ...
    八咫阅读 225评论 1 0