13 декабря 2022 1851
Долгое время для создания глубоких копий значений в JavaScript разработчикам приходилось использовать обходные решения и библиотеки. Сейчас в JavaScript появилась встроенная функция structuredClone() для глубокого копирования, которая в разы облегчила жизнь джавистов. Этот API был реализован во всех браузерах, например, в Firefox версии Firefox 94. Еще одно преимущество API — он работает в Node 17 и Deno 1.14. Сегодня мы расскажем, как прямо сейчас начать использовать эту функцию, не испытывая никаких неудобств.
Глубокое копирование в JavaScript с использованием структурных клонов

Поверхностное копирование

Копирование значения в JavaScript почти всегда поверхностное, а не глубокое. Другими словами, изменения глубоко вложенных значений будут видимы как в копии, так и в оригинале.

Одним из способов создания поверхностной копии в JavaScript является использование оператора расширения объекта...:

const myOriginal = {
someProp: "with a string value",
anotherProp: {
    withAnotherProp: 1,
    andAnotherProp: true
}
};
const myShallowCopy = {...myOriginal};

Добавление или изменение свойства непосредственно в поверхностной копии повлияет только на копию, но не на оригинал:


myShallowCopy.aNewProp = "a new value";
console.log(myOriginal.aNewProp)
// ^ logs `undefined`

Однако добавление или изменение глубоко встроенных свойств влияет как на копию, так и на оригинал:


myShallowCopy.anotherProp.aNewProp = "a new value";
console.log(myOriginal.anotherProp.aNewProp)
// ^ logs `a new value`

Выражение {...myOriginal} проводит итерацию по (перечислимым) свойствам myOriginal с помощью оператора Spread. При этом используется имя и значение свойства, которое присваивается одно за другим заново создаваемому, пустому объекту. В результате мы получаем объект, который будет идентичен по форме, но со своей собственной копией списка свойств и значений. Значения также копируются, но так называемые примитивные значения обрабатываются JavaScript не так, как непримитивные значения. Процитируем MDN:

В JavaScript примитив (примитивное значение, примитивный тип данных) — это данные, которые не являются объектом и не имеют методов. Существует семь примитивных типов данных: string, number, bigint, boolean, undefined, symbol и null.

MDN — Примитив

Непримитивные значения обрабатываются как ссылки, т.е. копирование значения на самом деле это просто копирование ссылки на один и тот же базовый объект, что и приводит к поверхностному копированию.

11235921_11104.jpg

Глубокое копирование

Противоположность поверхностного копирования — глубокое копирование. Алгоритм его работы предполагает копирование свойств объекта одного за другим, но действует он рекурсивно, когда обнаруживает ссылку на другой объект, создавая также и копию этого объекта. Это чрезвычайно важно для того, чтобы два куска кода случайно не обращались к одному и тому же объекту и не управляли состоянием друг друга.

Раньше в JavaScript не было простого и удобного способа создания глубоких копий значения. Многие программисты полагались на сторонние библиотеки, например использовали функцию cloneDeep() библиотеки Lodash. Пожалуй, самым популярным решением этой проблемы было использование хака на основе JSON:

const myDeepCopy = JSON.parse(JSON.stringify(myOriginal));

Это обходное решение было настолько популярным, что в
движке JavaScript V8 был оптимизирован метод JSON.parse() и в особенности указанный выше паттерн,
чтобы сделать его работу максимально быстрой. Он действительно работает быстро,
но имеет ряд недостатков и подводных камней:
  • Рекурсивные структуры данных. Метод JSON.stringify() работает, когда ему задана рекурсивная структура данных. Это часто происходит при работе со связанными списками или деревьями.
  • Встроенные типы. Метод JSON.stringify() работает, если значение содержит другие встроенные объекты JS, такие как MapSetDateRegExp или ArrayBuffer.
Функции. Метод JSON.stringify() отклоняет функции.

Структурное клонирование

Платформа уже давно нуждалась в возможности создавать глубокие копии значений JavaScript в ряде случаев: для хранения значения JS в IndexedDB требуется определенная сериализация, чтобы значение могло храниться на диске, потом десериализовано для восстановления значения JS. Точно так же для отправки сообщений в WebWorker с помощью postMessage() требуется перенести значение JS из одной области JS в другую. Используемый для этого алгоритм называется «структурный клон», и до недавнего времени он был практически недоступен для разработчиков.

Сейчас ситуация изменилась. Спецификация HTML была изменена так, чтобы выдавать функцию structuredClone(), которая выполняет именно этот алгоритм, нужный разработчикам для простого создания глубоких копий значений JavaScript.

Только и всего! Это весь API. Если вы хотите узнать подробности, посмотрите статью MDN.

Возможности и ограничения

Структурное клонирование устраняет многие (хотя и не все) недостатки метода JSON.stringify(). Структурное клонирование может обрабатывать циклические структуры данных, поддерживает многие встроенные типы данных и в целом работает более надежно и, как правило, быстрее.

Однако у него есть некоторые ограничения, которые могут стать для вас неожиданностью:

  • Прототипы. Если вы используете structuredClone() с экземпляром класса, то в качестве возвращаемого объекта вы получает основной объект, так как структурное клонирование отменяет цепочку прототипов объекта.

  • Функции. Если ваш объект содержит функции, они будут тихо удалены.

  • Неклонируемые значения. Для некоторых значений невозможно применить структурное клонирование, прежде всего для Error и узлам DOM. Попытка приведет к тому, что structuredClone() выдаст ошибку.

Если какие-либо из этих ограничений вам мешают, библиотеки, вроде Lodash, по-прежнему обеспечивают кастомную реализацию других алгоритмов глубокого копирования, которые могут подойти для вашего случая.

Производительность

Я не проводил нового сравнения показателей, но сравнивал разные методы в начале 2018 еще до появления structuredClone(). Тогда метод JSON.parse() был самым быстрым вариантом для очень маленьких объектов. Полагаю, что и сейчас это так. Методы, основанные на структурном клонировании, были (существенно) быстрее для более крупных объектов. Учитывая, что новый метод structuredClone() не нарушает работу других APIs и надежнее, чем JSON.parse(), я бы рекомендовал сделать его стандартным методом для глубокого копирования.

Заключение

Если вам нужно создать глубокую копию значения JS (например, потому что вы используете неизменяемые структуры данных или хотите, чтобы функция работала с объектом, не затрагивая оригинал), то вам больше не нужно искать обходные решения или библиотеки. Теперь в экосистеме JS есть structuredClone(). Ура!

Погрузитесь в изучение Java и освойте новые технологии на продвинутом уровне со скидкой 10%: https://ibs-training.ru/training/katalog_kursov/razrabotka_po_java/

Источник: https://web.dev/structured-clone/



Последние статьи в блоге

В Учебном центре IBS планируется запуск курсов по продуктам TData

Читайте о стратегическом соглашении TData и IBS и наших новых курсах

11 июня 2025

Компетенции бизнес-аналитиков: Junior и Middle в сравнении

В условиях динамично развивающейся ИТ-индустрии важно чётко понимать, какие навыки и знания необходимы для успешной работы на каждом этапе карьерного пути. Сегодня обсудим разницу в компетенциях ИТ бизнес-аналитиков уровней Junior и Middle. Если вы только начинаете свой путь в ИТ бизнес-анализе или, наоборот, уже обладаете некоторым опытом, этот материал поможет вам понять, какие навыки необходимы на каждом уровне и как развиваться дальше.

Новости
05 июня 2025

Лимит на сбои. Как понять, что система перегружена, а не просто плохо сделана?

Оценить производительность системы непросто, а контролировать еще сложнее. Как сделать так, чтобы внедряемая или уже эксплуатируемая система справлялась с нагрузками? Можно ли в этом вопросе полностью положиться на разработчиков ПО или вендоров? И кто в итоге будет отвечать за все простои системы? Рассказывает Николай Марченко, директор отделения нагрузочного тестирования компании IBS. Начать следует с того, что разбираться с последствиями возможных сбоев в любом случае придется тем, кто работает непосредственно с системой. Поэтому о вопросах производительности лучше задуматься еще на этапе внедрения.

Новости
03 июня 2025

Кто такой аналитик 1С?

Аналитик 1С — это специалист, который занимается оптимизацией бизнес-процессов с использованием программного обеспечения 1С. Его задача — анализировать существующие процессы, выявлять недостатки и разрабатывать решения для повышения эффективности. Аналитик переводит бизнес-требования в технические задания для разработчиков, становясь мостом между пользователями и ИТ-отделом. Эта роль требует как технических знаний, так и навыков коммуникации, что позволяет вносить значительный вклад в развитие компании и улучшение её конкурентоспособности.

Новости
28 мая 2025

Разбор задачи: UML-диаграмма классов для системы регистрации на курсы

Несколько дней назад мы предложили вам решить задачу — спроектировать диаграмму классов для системы регистрации студентов на учебные курсы в университете. Сегодня публикуем один из возможных вариантов решения. Сравните его с вашим и оцените, какие элементы вы отразили верно, а где можно усилить проработку.

22 мая 2025

Бизнес-аналитик и системный аналитик в ИТ: кто есть кто и в чем разница

Современные ИТ-проекты — будь то корпоративные решения, мобильные приложения или интеграционные платформы — требуют точного понимания как бизнес-целей, так и технических ограничений. На пересечении этих задач появляются две ключевые роли: бизнес-аналитик (БА) и системный аналитик (СА). Несмотря на схожесть направлений деятельности, эти специалисты действуют на разных уровнях и выполняют разные функции. Рассмотрим, кто они, каковы их зоны ответственности, чем они похожи, а чем принципиально отличаются.

21 мая 2025

5 распространенных ошибок в работе системных аналитиков

Ошибки системных аналитиков редко видны сразу, но последствия могут быть весьма заметными. Срыв сроков, недовольство заказчика, бесконечные правки требований, ощущение, что проект «расползается» — это часто не проблема менеджмента, а не выявленные вовремя аналитические ошибки и риски. Мы регулярно анализируем дипломные проекты выпускников курса «Системный аналитик» — не ради оценок, а чтобы понять, какие трудности реально возникают на практике, и обозначить направления для дальнейшего развития навыков. Даже у мотивированных специалистов с практическим опытом есть «слепые» зоны. Где-то не хватает чёткости в декомпозиции, где-то — качества проработки связей между сущностями, понимания архитектуры. Даже отсутствие умения аргументировать выбор решений перед бизнесом может негативно повлиять на проект. Мы вместе с Екатериной Тихомировой — практикующим аналитиком с более чем десятилетним опытом — разобрали некоторые типичные ошибки и риски, и способы, как их предотвратить.

20 мая 2025

Итоги работы Центра сертификации IBS

Центр сертификации IBS начал свою работу в апреле 2023 года, поэтому мы традиционно подводим итоги работы в апреле-мае. Прошедший год стал для нас периодом важных изменений. В 2024 году произошло несколько знаковых событий: наша команда обновила программы сертификации системных аналитиков и Java-разработчиков, подготовила к запуску сертификацию бизнес-аналитиков, получила аккредитацию от АПКИТ и стала обладателем Гран-при премии «Смарт пирамида». Рассказываем подробнее, каких результатов мы достигли в уходящем году и как это отразилось на нашей работе.

Новости
19 мая 2025

Какой метод тестирования выбрать: черный, белый или серый ящики?

При разработке программного обеспечения важно правильно подобрать методы тестирования. Это ключевой шаг для того, чтобы гарантировать высокое качество продукта. Понимание различных подходов, таких как тестирование черного, белого и серого ящика, помогает специалистам лучше находить ошибки и улучшать функциональность. Рассмотрим особенности, преимущества и варианты использования этих подходов.

14 мая 2025

Удостоверение, диплом и сертификат: в чем разница и что выбрать

В условиях стремительного развития технологий и постоянных изменений в бизнесе профессиональное развитие стало необходимостью. Особенно это актуально для специалистов в сфере IT и смежных отраслях. После прохождения курсов в учебных центрах слушатели могут получить один из трех типов документов: удостоверение о повышении квалификации, диплом о профессиональной переподготовке или сертификат. Несмотря на внешнюю схожесть, эти документы различаются по юридической силе, назначению и значимости для карьеры. Разберемся в этом подробнее.

12 мая 2025

Выгодный май — на курсы залетай!

Друзья, спешим поделиться отличной новостью — вы можете получить скидки до 40% на наши популярные курсы. Это отличная возможность улучшить навыки и инвестировать в профессиональное развитие по более выгодной цене. Выбирайте направление и подавайте заявку прямо сейчас!

05 мая 2025

Кейс: кастомизация курса по Jira

Кейс по проведению кастомизированного курса «Основы Jira» для крупной российской компании, занимающейся производством цифровой техники.

05 мая 2025

Зачем специалистам по 1С изучать системный анализ и архитектуру ПО

Как системный анализ и архитектура ПО помогают эффективнее работать в 1С.

29 апреля 2025

Банка Nutella, IT, ESG — что общего?

Когда вы читали этикетку на продукте не из-за состава, а из-за ESG-маркировки?

25 апреля 2025

Каковы плюсы и минусы монолитной и микросервисной архитектуры при разработке ИТ-продуктов?

Монолитная и микросервисная архитектуры представляют собой два различных подхода к разработке ИТ-продуктов, каждый из которых имеет свои преимущества и недостатки.

25 апреля 2025

Станьте архитектором ПО с выгодой! Только в апреле сэкономьте 20 000 ₽ и получите новый модуль по микросервисам в подарок

24 апреля стартует обучение на комплексной программе «Архитектор ПО. Путь к мастерству в проектировании систем»*.

14 апреля 2025

Архитектурные ошибки в корпоративных системах, которые могут создать проблемы в долгосрочной перспективе

В основе любой информационной системы — сложная структура. Спроектировать ее не легче, чем построить дом. Какие ошибки можно допустить при планировании ИТ-архитектуры, и в чем их основные причины? К чему они могут привести в долгосрочной перспективе и как этого избежать? Рассказывает Михаил Рощин, заместитель директора отделения управления проектами и архитектуры IBS.

Новости
10 апреля 2025

Кейс: Интенсив по управлению проектами для промышленной компании

Мы адаптировали курс по управлению проектами под запрос команды крупной промышленной компании и провели обучение. Вот что из этого вышло.

27 марта 2025

Кейс: Обучение сотрудников крупной компании работе с ClickHouse

Рассказываем, как мы организовали обучение команды работе с этой новой для заказчика технологией и каких результатов достигли.

19 марта 2025

Платформа сертификации IBS получила аккредитацию АПКИТ

Ассоциация предприятий компьютерных и информационных технологий (АПКИТ) приняла новый регламент сертификации ИТ-специалистов.

Новости
10 марта 2025

Не нашли, что искали? — Просто напишите, и мы поможем

Корпоративное обучение Оценка персонала Сертификация О нас Стань тренером Блог
Пользователь только что записался на курс ""
Спасибо!
Форма отправлена успешно.