JS之深拷贝探究

JS中的数据类型

基础数据类型

number,string,boolean,undefined,null,symbol这种,基本数据类型都是直接存储在内存中的内存栈上的,数据本身的值就是存储在栈空间里面

引用数据类型

object。存储在内存栈上的是指针,指向内存堆中的对象本身。
基本数据类型和引用类型的区别主要在于基本数据类型是分配在栈上的,而引用类型是分配在堆上的

实现方法

序列反序列

JSON.parse(JSON.stringify())的原理就是将源数据序列化称JSON字符串,然后再解析JSON字符串,构造由字符串描述的JavaScript值或对象。当拷贝JSON字符串的时候会开辟一个新的内存地址,从而完全切断和源数据的联系。

  • 优势:在开发中只考虑JSON对象(string, number, 对象, 数组, boolean或 null)的话用这个就可以了。

  • 存在的问题:

  1. undefined、Symbol和function字段会丢失;
  2. RegExp和Error会变成空对象{};
  3. Date对象会变成字符串;
  4. NaN、Infinite和-Infinite都会变转成null;
  5. 如果是由构造函数实例化出来的函数,原型上的constructor也会消失
  6. 如果对包含循环引用的对象(对象之间相互引用,形成无限循环)执行此方法,会抛出错误

lodash的cloneDeep

  • 优势:
  1. 可以拷贝循环引用(闭环)的对象;
  2. 可以拷贝ES6 引入的大量新的标准对象
/** `Object#toString` result references. */
var argsTag = '[object Arguments]',
    arrayTag = '[object Array]',
    boolTag = '[object Boolean]',
    dateTag = '[object Date]',
    errorTag = '[object Error]',
    funcTag = '[object Function]',
    mapTag = '[object Map]',
    numberTag = '[object Number]',
    objectTag = '[object Object]',
    regexpTag = '[object RegExp]',
    setTag = '[object Set]',
    stringTag = '[object String]',
    weakMapTag = '[object WeakMap]';

var arrayBufferTag = '[object ArrayBuffer]',
    float32Tag = '[object Float32Array]',
    float64Tag = '[object Float64Array]',
    int8Tag = '[object Int8Array]',
    int16Tag = '[object Int16Array]',
    int32Tag = '[object Int32Array]',
    uint8Tag = '[object Uint8Array]',
    uint8ClampedTag = '[object Uint8ClampedArray]',
    uint16Tag = '[object Uint16Array]',
    uint32Tag = '[object Uint32Array]';
  • 不足:
  1. 不能拷贝自身不可枚举类型的属性(enumerable);
  2. 不能拷贝普通数组的属性,除了RegExp.exec()返回的数组(只能复制出input和index属性,groups也不能复制);
  3. 不能拷贝BOM对象以及DOM对象(用cloneDeepWith)

迭代递归

for...in循环是 遍历对象的每一个可枚举属性,包括原型链上面的可枚举属性,

Object.keys()只是遍历自身的可枚举属性,不可以遍历原型链上的可枚举属性.

而Object.getOwnPropertyNames()则是遍历自身所有属性(不论是否是可枚举的),不包括原型链上面的。

  • 拷贝原型:Object.create(proto, [propertiesObject])
  • 拷贝不可枚举属性:Object.getOwnPropertyDescriptors(obj)
  • 拷贝symbol属性:Object.getOwnPropertySymbols(obj)
  • 拷贝循环引用属性: 借助hash表
function isObject(obj) {
    if ((typeof (obj) == 'object' || typeof (obj) == 'function') && obj !== null) {
        return true;
    }
}

function deepClone(obj, hash = new WeakMap()) {
    if (!isObject(obj)) {
        return obj
    }
    let tempObj;
    switch (obj.constructor) {
        case RegExp:
            tempObj = new RegExp(obj);
            break;
        case Date:
            tempObj = new Date(obj);
            break;
        case Function:
        case Error:
            tempObj = obj;
            break;
        default:
            // 查询哈希表,防止循环属性(环)
            if (hash.has(obj)) {
                return hash.get(obj);
            }
            // 初始化拷贝对象
            tempObj = Array.isArray(obj) ? [] : {};
            // 获取源对象所有属性描述符,获取到的value属于浅拷贝来的
            let allDesc = Object.getOwnPropertyDescriptors(obj);

            // 创建对象,拷贝原型,拷贝不可枚举属性
            tempObj = Object.create(Object.getPrototypeOf(obj), allDesc);

            // 获取原对象全部symbol属性
            let symKeys = Object.getOwnPropertySymbols(obj);
            // 拷贝symbol属性
            if (symKeys.length > 0) {
                symKeys.forEach(symKey => {
                    if (isObject(obj[symKey])) {
                        hash.set(obj, tempObj);
                        tempObj[symKey] = deepClone(obj[symKey], hash);
                    } else {
                        tempObj[symKey] = obj[symKey];
                    }
                })
            }

            // 拷贝可枚举属性
            Object.getOwnPropertyNames(obj).map(key => {
                if (isObject(obj[key])) {
                    hash.set(obj, tempObj);
                    tempObj[key] = deepClone(obj[key], hash);
                } else {
                    tempObj[key] = obj[key];
                }
            })
    }
    return tempObj;
}
let person2 = deepClone(person1);
console.log("person1:", person1);
console.log("person2:", person2);
function Person(name, age) {
    this.name = name;
}
let sym = Symbol('我是一个Symbol');
Person.prototype = {
    constructor: Person,
    shape: 'circle', //设置原型上可枚举
    [sym]: 'symbol' //设置原型上的 Symol 类型键
}

//设置原型上不可枚举
Object.defineProperty(Person.prototype, 'year', {
    value: 55,
    enumerable: false
});

var person1 = new Person("Lily");
// 设置自身子对象
person1.obj = {
    name: '我是一个对象',
    id: 1,
    object: {
        name: '子对象',
        id: '20181215',
        subObject: {
            name: '子对象的子对象',
            id: '18:37'
        }
    }
};
//设置自身不可枚举
Object.defineProperty(person1, 'year2', {
    value: {
        name: 'year2',
        id: '2117'
    },
    enumerable: false,
    writable: true,
    configurable: true,
});

//设置自身 Symol 类型键
let sym2 = Symbol('我是一个Symbol');
let sym3 = Symbol('我是一个Symbol');
person1[sym2] = {
    name: 'symbol2',
    id: 'id123',
    [sym3]: '内嵌symbol'
};
// 设置自身循环引用
person1.loopObj = person1;

let person2 = deepClone(person1);
// 修改person1的symbole属性
Object.getOwnPropertySymbols(person1).forEach(symKey => {
    person1[symKey] = '新的symbol值'
})

// ----------------------------------------------
// 修改person1的不可枚举属性
Object.defineProperty(person1, 'year2', {
    value: '新的不可枚举值',
});
// 或者
// person1.year2 = '新的不可枚举值'
console.log("person1.year2:", person1.year2);
// ----------------------------------------------

console.log("person1:", person1);
console.log("person2:", person2);
console.log("Person.prototype.isPrototypeOf(person2):", Person.prototype.isPrototypeOf(person2));
console.log("person1.year2===person2.year2:", person1.year2 === person2.year2)
console.log("person1[sym2]===person2[sym2]:", person1[sym2] === person2[sym2])
console.log("person1.obj.object.subObject===person2.obj.object.subObject:", person1.obj.object.subObject ===
    person2.obj.object.subObject)

效率比较:

var x = {};
for (var i = 0; i < 1000; i++) {
    x[i] = {};
    for (var j = 0; j < 1000; j++) {
        x[i][j] = Math.random();
    }
}

var start = Date.now();
var y = clone(x);
console.log(Date.now() - start);
深复制方法 JSON.parse _.cloneDeep deepClone
耗时 654 213 893

总结:总的来说没有一个方法是放诸四海而皆准的,建议使用 lodash的cloneDeep,但是对于小程序来说还是大了点,所以我平常开发小程序就是用JSON.parse(JSON.stringify())。

参考

  1. Javascript继承机制的设计思想
  2. JavaScript之对象序列化详解
  3. javaScript中浅拷贝和深拷贝的实现
  4. 深入剖析 JavaScript 的深复制
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,076评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,658评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,732评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,493评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,591评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,598评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,601评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,348评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,797评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,114评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,278评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,953评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,585评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,202评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,180评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,139评论 2 352

推荐阅读更多精彩内容