SOFTWARE DEVELOPMENT | 4 mins read

จัดการ Array ด้วย Javascript (Clone Deep)

By Son on 06 Jan 2021
sennalabs-blog-banner

ในปัจจุบันนี้ ปฏิเสธไม่ได้เลยว่าภาษาที่ถูกใช้ในการเขียนเว็บต่าง ๆ นั้น คงหนีไม่พ้นภาษา Javascript ซึ่งเป็นภาษาที่ถูกนำไปพัฒนาเป็น framework หรือ library ต่าง ๆ มากมาย 

ผู้พัฒนาหลายคนก็มีรูปแบบการเขียนภาษา Javascript ที่แตกต่างกัน เราเลยมีแนวทางการเขียนที่หลากหลาย มาแบ่งปันเพื่อน ๆ เกี่ยวกับการจัดการ Array ด้วยภาษา Javascript กัน เรามาดูตัวอย่างกันเลยดีกว่า

โดยปกติแล้วการ copy ค่าจาก value type ธรรมดา สามารถเขียนได้ดังนี้ let value = 3;

let valueCopy = value; // create copy
                console.log(valueCopy); // 3
                valueCopy = 100 // Change valueCopy
                console.log(valueCopy); // 100// ✅ Original NOT affected 
                console.log(value); // 3ซึ่งค่าของ value จะไม่เกิดผลกระทบเมื่อค่าที่ถูก copy ไปถูกเปลี่ยนแปลงแต่ การ copy ค่าจาก array type นั้นไม่เหมือนกับ value type ดังตัวอย่าง
                let array = [1,2,3];
                let arrayCopy = array; // create copy
                console.log(arrayCopy); // [1,2,3];
                arrayCopy[0] = '👻'; // Change 1st element of the array
                console.log(arrayCopy); // [ '👻', 2, 3 ] // ❌Original got affected
                console.log(array); // [ '👻', 2, 3 ] 

 

จะเห็นว่า ค่าของ original array นั้นถูกเปลี่ยนค่าด้วย เมื่อทำการ copy วิธีเดียวกับ value type เนื่องจาก สิ่งที่ copy มานั้น ไม่ใช่ค่า array แต่เป็น address ที่ถูกชี้ไปยังพื้นที่หน่วยความจำของ array ที่ใช้ เมื่อเราทำการ copy ค่ามา นั่นหมายถึงตัวแปรที่ copy มาจะมีพื้นที่หน่วยความจำเดียวกัน เมื่อค่าเปลี่ยน จึงทำให้ค่า origin array ถูกเปลี่ยนค่าด้วยทันที

วีธีการ clone  ค่าจาก array ที่ถูกต้อง

ดังนั้นวีธีการ clone หรือ copy ค่าจาก array ที่ถูกต้องนั้น สามารถเขียนง่าย ๆ ดังนี้let array = [1,2,3];

let arrayCopy = [...array]; // create TRUE copy
                console.log(arrayCopy); // [1,2,3];
                arrayCopy[0] = '👻'; // Change 1st element of the array
                console.log(arrayCopy); // [ '👻', 2, 3 ] // ✅ Original NOT affected 
                console.log(array); // [ 1, 2, 3 ] แต่ !!! การใช้ [...] spread operator ไม่สามารถใช้ได้ทุกกรณี สามารถใช้ได้ฉพาะ array ที่มี level เดียวเท่านั้น ยกตัวอย่างหากเป็น nestedArray ดังตัวอย่าง
                let nestedArray = [1, [2], 3];
                let arrayCopy = [...nestedArray3];// Make some changes
                arrayCopy[0] = '👻'; // change shallow element
                arrayCopy[1][0] = '💩'; // change nested element
                console.log(arrayCopy); // [ '👻', [ '💩' ], 3 ]// ❌ Nested array got affected
                console.log(nestedArray); // [ 1, [ '💩' ], 3 ] จะเห็นว่าค่าของ origin nestedArray ถูกเปลี่ยนแปลงค่าด้วยเช่นกัน ดังนั้นวิธีที่ปลอดภัยที่สุดมีดังนี้หากเป็น 
                nestedArraylet nestedArray = [1, [2], 3];
                let arrayCopy = JSON.parse(JSON.stringify(nestedArray));
                // Make some changes
                arrayCopy[0] = '👻'; // change shallow element
                arrayCopy[1][0] = '💩'; // change nested element
                console.log(arrayCopy); // [ '👻', [ '💩' ], 3 ]
                // ✅ Nested array NOT affected
                console.log(nestedArray); //  1, [ 2 ], 3 ]

 

ซึ่งวิธีนี้จะทำการเปลี่ยน array ให้เป็น string ด้วย JSON.stringify และเปลี่ยนกลับเป็น array อีกครั้งด้วย JSON.parse โดยจะได้ค่า array จริง ๆ แต่วิธีนี้จะไม่สามารถใช้ได้ทุกกรณีเช่นกันหาก array มีฟังก์ชัน หรือสัญลักษณ์ต่าง ๆ

ความต่างระหว่าง JSON.stringify / parse กับ deepClone

จึงทำให้วิธีการ JSON.stringify / parse กับ วิธีการ deepClone ด้วย library อื่นมาช่วย นั้นต่างกันดังนี้

  • JSON.stringify / parse ใช้ได้กับตัวอักษร Number และ String และ Object เท่านั้นโดยไม่มีฟังก์ชันหรือคุณสมบัติ Symbol
  • deepClone ทำงานกับทุกประเภทฟังก์ชั่นและสัญลักษณ์จะถูกคัดลอกโดย reference

วิธีที่ clone array แบบง่าย ๆ

จากตัวอย่างที่กล่าวไปข้างต้น เรามีวิธีที่ช่วย clone array ที่ง่ายและขึ้นอยู่กับการใช้งาน ดังนี้

1. ใช้ JSON.parse(JSON.stringify(array)) (ใช้ได้กับตัวอักษร Number และ String และ Object)

2. ใช้  library อื่น ๆ ที่ช่วยในการจัดการ array เช่น _.cloneDeep ของ library lodash เป็นต้น (ใช้ได้ทุกกรณี)

// How to deep clone array const numbers = [1, [2], [3,[4],5]] // Using javascript JSON.parse(JSON.stringify(numbers)) // Using lodash _.cloneDeep(numbers)

Original article:

Written By
Son

Please Tell Us Your Ideas

We will get back to you within 24 hours!