일반 object clone

황제낙엽 2011.07.08 17:44 조회 수 : 2102

sitelink1 http://blog.imaginea.com/deep-copy-in-javascript/ 
sitelink2  
sitelink3  
sitelink4 http://1 
extra_vars4 ko 
extra_vars5  
extra_vars6 sitelink1 
 

Some of you might have come across this issue. When you make a copy of an object and try to change the values of the new instance, it reflects in the original one too. Well I heard this not only applies to JavaScript, but also to most other object oriented languages; not sure though.

Here is a case:

Problem
//original object definition
var oOriginal = {
    memNum: 1,                                        // number
    memStr: "I am a string",                          // string
    memObj: {
        test1: "Old value"                            // we'll test
    },
    memArr: [                                         // array with object members
        "a string",                                   // simple string element
        {                                             // an object
            test2: "Try changing me"                  // we'll test
        }
    ]
};

//test 1
var oCopy = oOriginal;                                // normal copy

oCopy.memObj.test1 = "New value";                     // Problem
                                                      // - will reflect in oOriginal

alert(oOriginal.memObj.test1);                        // will show "New value"

//test 2
oCopy.memArr[1].test2 = "I am changed";               // Problem
                                                      // - will reflect in oOriginal

alert(oOriginal.memArr[1].test2);                     // will show "I am changed"

 

 

Here the problem is, the objects are never copied, only their references are. So what’s the solution? Iterate through all members of the original object, create the same members for target object and then assign corresponding values. And while doing so, we can’t forget array of objects.

We’ll add this simple line at the very top to help us detect Array objects. It basically adds our own custom extension to the Array class.

To recognize Array type objects
<html>
<head>
<title>Deep Copy in JavaScript</title>
<script language="javascript" type="text/javascript">
    Array.prototype.__isArray = true;
</script>
.
.

 

 

Now, here is the generic method that will help in iterating through all (type of) members of our original object and copy them one-by-one regardless of the depth of the object structure:

Solution
var ObjectHandler = {
    //public method
    getCloneOfObject: function(oldObject) {
        var tempClone = {};

        if (typeof(oldObject) == "object")
            for (prop in oldObject)
                // for array use private method getCloneOfArray
                if ((typeof(oldObject[prop]) == "object") &&
                                (oldObject[prop]).__isArray)
                    tempClone[prop] = this.getCloneOfArray(oldObject[prop]);
                // for object make recursive call to getCloneOfObject
                else if (typeof(oldObject[prop]) == "object")
                    tempClone[prop] = this.getCloneOfObject(oldObject[prop]);
                // normal (non-object type) members
                else
                    tempClone[prop] = oldObject[prop];

        return tempClone;
    },

    //private method (to copy array of objects) - getCloneOfObject will use this internally
    getCloneOfArray: function(oldArray) {
        var tempClone = [];

        for (var arrIndex = 0; arrIndex <= oldArray.length; arrIndex++)
            if (typeof(oldArray[arrIndex]) == "object")
                tempClone.push(this.getCloneOfObject(oldArray[arrIndex]));
            else
                tempClone.push(oldArray[arrIndex]);

        return tempClone;
    }
};

 

 

Now we’ll perform the same two tests mentioned in the ‘Problem’ block using this new helper – ObjectHandler

Test
//test 1
var oCopy = ObjectHandler.getCloneOfObject(oOriginal);

oCopy.memObj.test1 = "New value";

alert(oOriginal.memObj.test1);                        // will show "Old value"

//test 2
oCopy.memArr[1].test2 = "I am changed";

alert(oOriginal.memArr[1].test2);                     // will show "Try changing me"

 

 

In general, this solution is often referred to as "Deep Copy".

번호 제목 글쓴이 날짜 조회 수
126 Function.apply() 와 Function.call() file 황제낙엽 2011.10.07 363
125 String 에 trim() 함수 적용하기 황제낙엽 2011.08.28 407
124 javascript array contains method 황제낙엽 2011.08.19 519
123 daumopeneditor 황제낙엽 2011.07.18 372
» object clone 황제낙엽 2011.07.08 2102
121 YUI Compressor in Java file 황제낙엽 2011.07.04 343
120 iframe auto resize (cross browsing) 황제낙엽 2011.05.13 982
119 자바스크립트 정렬(javascript formatter) 도구 JavaScript Code Improver file 황제낙엽 2011.04.04 376
118 Why does this simple Javascript prototype not work in IE? 황제낙엽 2011.03.24 630
117 Defining classes and inheritance (클래스 정의와 상속) 황제낙엽 2011.03.24 642
116 User Agent 관련 Reference URL 황제낙엽 2011.02.22 398
115 각 브라우저 별 User Agent 정보 황제낙엽 2011.02.22 1176
114 History of User Agent 황제낙엽 2011.02.22 409
113 Navigator 객체란? 황제낙엽 2011.02.22 454
112 Understanding User-Agent Strings 황제낙엽 2011.02.22 444
111 User Agent 정보 모음 file 황제낙엽 2011.02.22 10679
110 ActiveX 설치 여부를 검사하는 스크립트 황제낙엽 2011.02.13 4400
109 [JavaScript Tutorials] Error handling in JavaScript using try/catch/finally - The Error object and throwing your own errors (해석중) 황제낙엽 2009.04.10 386
108 [JavaScript Tutorials] More leakage patterns (해석중) 황제낙엽 2009.04.10 402
107 [JavaScript Tutorials] Introducing the closure (해석중) 황제낙엽 2009.04.10 912