zoukankan      html  css  js  c++  java
  • javascript对象深拷贝

    JSON.stringify()以及JSON.parse()

    优点:简易
    缺点:使用JSON.stringify()以及JSON.parse()它是不可以拷贝 undefined , function, RegExp 等等类型的
    当然对于函数可以使用JSON.stringify()和JSON.parse()的第二个参数进行转换

    function stringifyRep(key, value) {
        if (typeof value === "function") {
            return `${value}`;
        }
        return value;
    }
    function parseRep(key, value) {
        return eval(value);
    }
    var a = {
        b: () => 1 + 1
    }
    var aa = JSON.parse(JSON.stringify(a, stringifyRep), parseRep)
    

    Object.assign(target, source)

    Object.assign({}, obj)
    

    仅可拷贝第一层

    扩展符 {...obj}

    仅可拷贝第一层

    数组中的slice() & concat()

    var arr1 = [1,2,3]
    var arr2 = arr1.slice()  // 方法一
    // var arr2 = arr1.concat()  //方法二
    arr2.push(4)
    console.log('arr1:', arr1)  // arr1: [1, 2, 3]
    console.log('arr2:', arr2)  // arr1: [1, 2, 3, 4]
    

    第三方库

    jQuery.extend 和 lodash.cloneDeep

    递归拷贝——最优解决方案

    简易版递归深拷贝

    function deepClone(obj) {
        if(obj === null) return null;
        if(obj instanceof RegExp) return new RegExp(obj);
        if(obj instanceof Date) return new Date(obj);
        if(typeof obj !== 'object') {
            return obj;
        }
        let t = new obj.constructor();
        for(let key in obj) {
            t[key] = deepClone(obj[key]);
        }
        return t;
    }
    

    vuex中的深拷贝代码

    vuex中深拷贝代码考虑了循环引用等多种问题,更加完善,且写的简明,学习一下:
    https://github.com/vuejs/vuex/blob/dev/src/util.js#L22

    export function find (list, f) {
      return list.filter(f)[0]
    }
    /**
     * 深拷贝传入的对象:
     * 缓存所有嵌套对象和它们的拷贝对象。
     * 如果传入对象为循环引用结构,利用缓存过的拷贝避免无限循环。
     * 
     * @param {*} obj
     * @param {Array<Object>} cache
     * @return {*}
     */
    export function deepCopy (obj, cache = []) {
      // just return if obj is immutable value
      if (obj === null || typeof obj !== 'object') {
        return obj
      }
    
      // 检测obj是否是循环引用结构,是的话直接返回它的拷贝对象
      const hit = find(cache, c => c.original === obj)
      if (hit) {
        return hit.copy
      }
    
      const copy = Array.isArray(obj) ? [] : {}
      // 首先将copy放入cache, 因为我们需要在递归deepCopy的时候引用它
      cache.push({
        original: obj,
        copy
      })
    
      Object.keys(obj).forEach(key => {
        copy[key] = deepCopy(obj[key], cache)
      })
    
      return copy
    }
    
  • 相关阅读:
    Kafka概述与部署
    理解微前端技术原理
    Docker之单机网络一
    MySQL Binlog解析
    MySQL之Xtrabackup
    MySQL之mysqldump
    学生成绩管理系统
    凯域课程表使用体验(软件名:上课了)
    动手动脑问题分析
    关于十天集训的收获
  • 原文地址:https://www.cnblogs.com/emilyzhou/p/13601992.html
Copyright © 2011-2022 走看看