일반 object clone

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

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".

번호 제목 글쓴이 날짜 조회 수
146 외부 라이브러리 (.js) 의 바람직하지 않은 동적 로딩 (eval함수 이용) 황제낙엽 2012.01.18 2195
145 IFrames and cross-domain security file 황제낙엽 2012.01.13 606
144 자바스크립트 숫자형 체크 함수 (isFinite() 함수 와 isNaN() 함수) 황제낙엽 2011.12.13 484
143 char to hex string 황제낙엽 2011.11.29 590
142 Understanding delete 황제낙엽 2011.11.28 458
141 JScript Memory Leaks 황제낙엽 2011.11.28 654
140 JavaScript Reserved Words 황제낙엽 2011.11.28 854
139 Reference Count (순환참조) 황제낙엽 2011.11.24 576
138 IE and Memory accumulation in Javascript (document.createElement()) file 황제낙엽 2011.11.24 1614
137 String Performance: Getting Good Performance from Internet Explorer (IE7) 황제낙엽 2011.11.24 717
136 Memory leak 및 성능 측정 도구 file 황제낙엽 2011.11.23 970
135 JavaScript Array.push Performance 황제낙엽 2011.11.21 724
134 CKEditor 3 JavaScript API Documentation 황제낙엽 2011.11.14 521
133 다양한 WYSIWYG 에디터 황제낙엽 2011.11.13 439
132 Faster JavaScript Memoization For Improved Application Performance 황제낙엽 2011.11.04 690
131 Top 10 JavaScript Performance Tips+Android & iPhone Engine Testing 황제낙엽 2011.11.04 432
130 url encode & decode 황제낙엽 2011.10.30 796
129 페이지 스크롤 끝 확인 황제낙엽 2011.10.24 6859
128 call() and apply() methods in Javascript 황제낙엽 2011.10.07 712
127 Function.apply and Function.call in JavaScript 황제낙엽 2011.10.07 620