zoukankan      html  css  js  c++  java
  • JavaScript:深拷贝和浅拷贝区别,以及实现深拷贝的方法

    区别:

    深拷贝和浅拷贝最根本的区别在于是否是真正获取了一个对象的复制实体,而不是引用,

    深拷贝在计算机中开辟了一块内存地址用于存放复制的对象,而浅拷贝仅仅是指向被拷贝的内存地址,如果原地址中对象被改变了,那么浅拷贝出来的对象也会相应改变。

    实现深拷贝的方法:

    • 最简单的方法就是JSON.parse(JSON.stringify())

    需要注意的是:这种拷贝方法不可以拷贝一些特殊的属性(例如正则表达式,undefine,function)

    • 用递归去复制所有层级属性
    function deepCopyTwo(obj) {
        let objClone = Array.isArray(obj) ? [] : {};
        if (obj && typeof obj == 'object') {
            for (const key in obj) {
                //判断obj子元素是否为对象,如果是,递归复制
                if (obj[key] && typeof obj[key] === "object") {
                    objClone[key] = deepCopyTwo(obj[key]);
                } else {
                    //如果不是,简单复制
                    objClone[key] = obj[key];
                }
            }
        }
        return objClone;
    }
    export const shallowClone = (object: any) => {
        const newObject = {};
        for (const key in object) {
            if (object.hasOwnProperty(key)) {
                newObject[key] = object[key];
            }
        }
        return newObject;
    }
    
    export const jsonClone = (object: any) => {
        return JSON.parse(JSON.stringify(object));
    }
    
    export const cloneDeep = (object: any) => {
        if (typeof object !== "object") return object;
        const newObject = object instanceof Array ? [] : {};
        for (const key in object) {
            if (object.hasOwnProperty(key)) {
                if (typeof object[key] === "object" && object[key] !== null) {
                    newObject[key] = cloneDeep(object[key]);
                } else {
                    newObject[key] = object[key];
                }
            }
        }
        return newObject
    }
    
    // 对特殊对象进行类型判断
    const isType = (object, type): boolean => {
        if (typeof object !== "object") return false;
        const typeString = Object.prototype.toString.call(object);
        switch (type) {
            case "Array":
                return typeString === "[object Array]";
            case "Date":
                return typeString === "[object Date]";
            case "RegExp":
                return typeString === "[object RegExp]";
            default:
                return false;
        }
    }
    
    // 实现一个提取flags的函数
    const getRegExpFlags = (regExp: RegExp) => {
        let flags = "";
        if (regExp.global) flags += "g";
        if (regExp.ignoreCase) flags += "i";
        if (regExp.multiline) flags += "m";
        return flags;
    }
    // Buffer对象、Promise、Set、Map暂未处理
    export const deepClone = oldObject => {
        // 维护两个储存循环引用的数组
        const oldObjects = [];
        const newObjects = [];
    
        const _deepClone = oldObject => {
            // 递归直到oldobject为null时,或类型不为“object”时停止。
            if (oldObject === null) return null;
            if (typeof oldObject !== 'object') return oldObject; // 
    
            let newObject, newProtoType;
    
            if (isType(oldObject, 'Array')) {
                /**
                 * 对数组做特殊处理
                 * 数组里面如果单纯只有多个基本数据类型将在判断是否时object对象时进行return
                 * 比如数组[1,4,6]
                 * forin循环里key为数组下标,进行循环复制,都止步于前面两个return,此时的循环不会进行定义新newObject
                 * new Array(2) [undefined, undefined],下面的forin循环不会对这个Array进行循环,
                 * 所以最后拷贝结果是[]不是[undefined, undefined]
                 */
                newObject = [];
            } else if (isType(oldObject, 'RegExp')) {
                // 对正则对象做特殊处理,下面的newObject[key] = 此次返回的newObject
                newObject = new RegExp(oldObject.source, getRegExpFlags(oldObject));
                if (oldObject.lastIndex) newObject.lastIndex = oldObject.lastIndex;
            } else if (isType(oldObject, 'Date')) {
                // 对Date对象做特殊处理
                newObject = new Date(oldObject.getTime());
            } else {
                // 处理对象原型
                newProtoType = Object.getPrototypeOf(oldObject);
                // 利用Object.create切断原型链
                newObject = Object.create(newProtoType);
            }
    
            // 处理循环引用
            const index = oldObjects.indexOf(oldObject);
    
            if (index != -1) {
                // 如果父数组存在本对象,说明之前已经被引用过,直接返回此对象
                return newObjects[index];
            }
            oldObjects.push(oldObject);
            newObjects.push(newObject);
    
            for (const key in oldObject) {
                if (oldObject.hasOwnProperty(key)) {
                    // newObject 已根据条件做了特殊处理
                    newObject[key] = _deepClone(oldObject[key]);
                }
            }
    
            return newObject;
        };
        return _deepClone(oldObject);
    };

    浅拷贝:object.assign(target,source)

  • 相关阅读:
    一个漂亮的lazarus做的pagecontrol
    预测,阿里盒子必将失败!
    sex在软件开发中的运用--SIX技术
    糟糕的@@identity,SCOPE_IDENTITY ,IDENT_CURRENT
    Delphi、Lazarus保留字、关键字详解
    糟糕的界面设计
    Firebird存储过程--更加人性化的设计
    lazarus的动态方法和虚拟方法
    用户行为导向的交互设计
    Javascript的一个怪现象
  • 原文地址:https://www.cnblogs.com/Nyan-Workflow-FC/p/12930986.html
Copyright © 2011-2022 走看看