preface

Object is one of the basic types in JS, and is closely related to prototype chain, array and other knowledge. Deep-copy objects come up in interviews and in real development.

As the name implies, deep copy is a complete copy of an object out of memory. So no matter what method is used, it is inevitable to open up a new memory space.

There are two common ways to implement deep copy:

  1. Iterative recursion
  2. Serialization Deserialization method

We will test and compare common implementations based on a test case:

let test = {
    num: 0,
    str: ' ',
    boolean: true,
    unf: undefined,
    nul: null,
    obj: {
        name: 'I am an object',
        id: 1
    },
    arr: [0, 1, 2],
    func: function() {
        console.log(I'm a function.)
    },
    date: new Date(0),
    reg: new RegExp('/ I'm a regular /ig'),
    err: new Error('I was a mistake')}let result = deepClone(test)

console.log(result)
for (let key in result) {
    if (isObject(result[key]))
        console.log(`${key}The same? `, result[key] ===test[key])} // Check whether it is an objectfunction isObject(o) {
    return (typeof o === 'object' || typeof o === 'function') && o ! == null }Copy the code

1. Iterative recursion

This is the most general approach, and the idea is simple: you iterate over an object, making a deep recursive copy of each of its values.

  • for… in 法

// Iterative recursion: deep copy objects and arraysfunction deepClone(obj) {
    if(! isObject(obj)) { throw new Error('obj is not an object! ')}let isArray = Array.isArray(obj)
    let cloneObj = isArray ? [] : {}
    for (let key in obj) {
        cloneObj[key] = isObject(obj[key]) ? deepClone(obj[key]) : obj[key]
    }

    return cloneObj
}
Copy the code
Results:Copy the code

We found that both ARR and OBJ were deep-copied successfully and their memory references were different, but func, Date, REg, and ERR were not copied successfully because they had special constructors.Copy the code
  • Reflect method

/ / agent methodfunction deepClone(obj) {
    if(! isObject(obj)) { throw new Error('obj is not an object! ')}let isArray = Array.isArray(obj)
    let cloneObj = isArray ? [...obj] : { ... obj } Reflect.ownKeys(cloneObj).forEach(key => {
        cloneObj[key] = isObject(obj[key]) ? deepClone(obj[key]) : obj[key]
    })

    return cloneObj
}
Copy the code
Results:Copy the code

We find that as a result and using for... In the same. So what are the advantages? Readers can take a guess, and we'll reveal the answer below.Copy the code
  • Deep copy in Lodash

    The cloneDeep method in LoDash is also implemented in this way, but it supports a wider variety of objects. For details, refer to LoDash’s baseClone method.

    We replaced the deep-copy function used in our test case with lodash’s:

    let result = _.cloneDeep(test)
    Copy the code

    Results:

    We found that arR, OBj, date, and REG deep copies were successful, but func and ERR memory references remained unchanged.

    Why doesn’t it change? That’s a question I’ll leave to the reader to explore, but it has something to do with cloneableTags in Lodash.

    Because there are so many types of objects in the front-end, LoDash also provides users with a custom deep-copy method cloneDeepWith, such as custom deep-copy DOM objects:

    function customizer(value) {
      if (_.isElement(value)) {
        return value.cloneNode(true); } } var el = _.cloneDeepWith(document.body, customizer); console.log(el === document.body); / / = >falseconsole.log(el.nodeName); / / = >'BODY'console.log(el.childNodes.length); / / = > 20Copy the code

Serialization Deserialization method

This method is interesting because it serializes code into data and then deserializes it back into objects:

// serialization deserializationfunction deepClone(obj) {
    return JSON.parse(JSON.stringify(obj))
}
Copy the code

Results:

Go deeper and deeper

  1. What if objects are looped? We add a loopObj key to test that points to itself:

    test.loopObj = test
    Copy the code

    In this case we use the first method for.. Both the in implementation and the Reflect implementation overflow the stack:

    The second method also returns an error:

    But Lodash gets it right:

    Why is that? Let’s take a look at the lodash source code:

    Because LoDash uses a stack to store objects, if there is a ring object, it will be detected in the stack and return the result directly. The idea is derived from the structured cloning algorithm defined by the HTML5 specification, and it also explains why LoDash does not copy Error and Function types.

    Of course, setting up a hash table to store copied objects serves the same purpose:

    function deepClone(obj, hash = new WeakMap()) {
        if(! isObject(obj)) {returnObj} //if (hash.has(obj)) return hash.get(obj)
    
        let isArray = Array.isArray(obj)
        let cloneObj = isArray ? [] : {} // hash. Set (obj, obj, obj)cloneObj)
    
        let result = Object.keys(obj).map(key => {
            return {
                [key]: deepClone(obj[key], hash)}})return Object.assign(cloneObj, ... result) }Copy the code

    Here we use WeakMap as a hash table because its keys are weakly referenced, and in our scenario the keys happen to be objects that need weak references.

  2. The key is not a string but a Symbol

    Let’s modify the test case:

    var test = {}
    let sym = Symbol('I'm a Symbol')
    test[sym] = 'symbol'
    
    let result = deepClone(test)
    console.log(result)
    console.log(result[sym] === test[sym])
    Copy the code

    Run for… Deep copy of the in implementation we will find:

    Copy failed. Why?

    Because Symbol is a special data type, its biggest characteristic is unique, so its deep copy is shallow copy.

    But if we use Reflect’s implementation at this point:

    It worked because for… In can’t get a Symbol key, while Reflect can.

    Sure, let’s revamp for… An in implementation can also:

    function deepClone(obj) {
        if(! isObject(obj)) { throw new Error('obj is not an object! ')}let isArray = Array.isArray(obj)
        let cloneObj = isArray ? [] : {}
        let symKeys = Object.getOwnPropertySymbols(obj)
        // console.log(symKey)
        if (symKeys.length > 0) {
            symKeys.forEach(symKey => {
                cloneObj[symKey] =  isObject(obj[symKey]) ? deepClone(obj[symKey]) : obj[symKey]
            })
        }
        for (let key in obj) {
            cloneObj[key] = isObject(obj[key]) ? deepClone(obj[key]) : obj[key]
        }
    
        return cloneObj
    }
    Copy the code
  3. Copy the properties on the prototype

    As we all know, JS objects are designed based on the prototype chain, so when an object’s property cannot be found it will be looked up its prototype chain, that is, the __proto__ property of a non-constructor object.

    We create a childTest variable with result as the result of its deep copy, all else unchanged:

    let childTest = Object.create(test)
    let result = deepClone(childTest)
    Copy the code

    At this point, the only four implementations we initially provided were for… The implementation of in copies correctly. Why? Again, the reason is in the structured cloning algorithm: attributes on the primitive chain are not tracked and copied.

    Fall on the concrete implementation is: for… In tracks properties on the prototype chain, while the other three methods (Object.keys, Reflect.ownKeys, and JSON methods) do not track properties on the prototype chain:

  4. Non-enumerable properties need to be copied

    In the fourth case, we need to copy non-enumerable properties like property descriptors, setters, and getters. Generally, this requires an additional set of non-enumerable properties to store them. Similar to using for… in the second case We define the property descriptors for the obj and arr properties in the test variable:

    Object.defineProperties(test, {
        'obj': {
            writable: false,
            enumerable: false,
            configurable: false
        },
        'arr': {
            get() {
                console.log('Get called')
                return[1, 2, 3]},set(val) {
                console.log('Set' is called)}}})Copy the code

    Then implement our copy version of the non-enumerable property:

    function deepClone(obj, hash = new WeakMap()) {
        if(! isObject(obj)) {returnObj} // table lookup to prevent circular copyif (hash.has(obj)) return hash.get(obj)
    
        letIsArray = array. isArray(obj) // Initializes the copy objectlet cloneObj = isArray ? [] : {} // hash. Set (obj, obj, obj)cloneObj) // Get all attribute descriptors of the source objectletAllDesc = Object. GetOwnPropertyDescriptors (obj) / / get all source Object Symbol type keyletSymKeys = Object. GetOwnPropertySymbols (obj)/copy/Symbol type key corresponding attributesif (symKeys.length > 0) {
            symKeys.forEach(symKey => {
                cloneObj[symKey] = isObject(obj[symKey]) ? deepClone(obj[symKey], hash) : obj[symKey]})} // Copy non-enumerable attributes, because allDesc value is a shallow copy, so it should be placed firstcloneObj = Object.create(
            Object.getPrototypeOf(cloneObj), allDesc) // Copy enumerable attributes (including those on the prototype chain)for (let key in obj) {
            cloneObj[key] = isObject(obj[key]) ? deepClone(obj[key], hash) : obj[key];
        }
    
        return cloneObj
    }
    Copy the code

    Results:

conclusion

  1. Daily deep copy, serialization deserialization method is recommended.
  2. Symol/Symol/Symol/Symol/Symol/Symol/Symol/Symol/Symol/Symol/Symol/Symol/Symol/Symol/Symol/Symol/Symol/Symol
// Encapsulate the deepClone function we wrote earlierfunction cloneDeep(obj) {
    let family = {}
    let parent = Object.getPrototypeOf(obj)

    while(parent ! = null) { family = completeAssign(deepClone(family), Parent) parent = object.getProtoTypeof (parent) https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Object/assignfunctioncompleteAssign(target, ... sources) { sources.forEach(source= > {let descriptors = Object.keys(source).reduce((descriptors, key) => {
                descriptors[key] = Object.getOwnPropertyDescriptor(source, key)
                returnDescriptors}, {}) / / Object. The default will assign copy enumerable Symbols Object. GetOwnPropertySymbols (source).forEach(sym => {
                let descriptor = Object.getOwnPropertyDescriptor(source, sym)
                if (descriptor.enumerable) {
                    descriptors[sym] = descriptor
                }
            })
            Object.defineProperties(target, descriptors)
        })
        return target
    }

    return completeAssign(deepClone(obj), family)
}

Copy the code
  1. For deep copies with special requirements, loDash’s copyDeep or copyDeepWith methods are recommended.

    No matter what you do, try not to complicate simple things. If deep copy can be used, it can be solved in a more elegant way, such as using a function to get objects. Of course, it is possible to install a force during the interview.