收藏文章 楼主

太卷了!浏览器也支持原生的深拷贝 API 了?

版块:网站建设   类型:普通   作者:小羊羔links   查看:612   回复:0   获赞:0   时间:2022-01-23 22:48:29

今天来聊一个前端老生常谈的话题,深拷贝。

在以前,由于浏览器并未对这个能力提供原生支持,所以它经常出现在 手写XXX 这样的面试题中,

不过,我们成功的把浏览器给卷了,现在它给我们提供了一个原生的深拷贝 API structuredClone

浅拷贝

再看深拷贝之前,我们还是先来简单回顾一下浅拷贝和深拷贝的区别(不要嫌我啰嗦,主要为了照顾新手同学,如果有基础的同学直接跳过 。

  • 浅拷贝 创建一个新对象,这个对象有着原始对象属性值的一份精确拷贝。如果属性是基本类型,拷贝的就是基本类型的值,如果属性是引用类型,拷贝的就是内存地址 ,所以如果其中一个对象改变了这个地址,就会影响到另一个对象。

  • 将一个对象从内存中完整的拷贝一份出来,从堆内存中开辟一个新的区域存放新对象,且修改新对象不会影响原对象。

在代码里我们复制一个对象可能用的最多的就是扩展运算符 ...,这就是一种浅拷贝。

const myOriginal = {
  someProp"code秘密花园",
  anotherProp: {
    withAnotherProp1,
    andAnotherProptrue
  }
};

const myShallowCopy = {...myOriginal};

我们直接在浅拷贝对象上添加 更改属性只会影响拷贝副本,而不会影响原始值

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

如果我们更改之前的引用对象属性,副本和原始值双方都会影响

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

本质上,就是拷贝对原始类型(string、number、bigint、boolean、undefined、symbol、null 和引用类型(objectarray 等 的处理不一样。

深拷贝

与浅拷贝相对的就是深拷贝,深拷贝算法也会一个一个地拷贝一个对象的属性,但是当它拷贝对另一个对象的引用时会递归调用,同时创建该引用类型的一个副本,这可以有效避免我们在代码里共享一个意想不到的对象引用。

在以前,我们需要依赖一些第三方库来实现深拷贝,比如 Lodash 的 cloneDeep() 函数, 者可能大多数人用的基于是 JSON 的 hack 

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

但是这种方法缺点很多

  • 循环引用 JSON.stringify() 的对象中如果有循环引用会抛出异常 Converting circular structure to JSON
  • 其他数据类型 JSON.stringify() 无法拷贝 Map、Set、RegExp 这些特殊数据类型。
  • 函数 JSON.stringify() 会默认移除函数。

structuredClone

现在,structuredClone API 已经成为了一个 HTML 规范中的标准提案,用它可以轻松实现一个深拷贝,并且也默认解决了循环引用等问题、支持了很多默认的数据类型。

// Create an object with a value and a circular reference to itself.
const original = { name"MDN" };
original.itself = original;

// Clone it
const clone = structuredClone(original);

console.assert(clone !== original); // the objects are not the same (not same identity)
console.assert(clone.name === "MDN"); // they do have the same values
console.assert(clone.itself === clone); // and the circular reference is preserved

并且,相比 JSON.parse() ,structuredClone API 的性能更好,特别是在处理一些更大复杂的对象的时候,所以我们可以用它来作为代码里深拷贝的默认方法啦,为了兼容性考虑,可以用 JSON.stringify  者其他工具函数作为备用。

不过,这个 API 也并不完美,它也有些缺点

  • 原型 无法拷贝对象的原型链。
  • 函数 无法拷贝函数。
  • 不可克隆 并没有支持所有类型的拷贝,比如 Error

当然,大部分实际的需求场景中,我们没必要拷贝这些东西,估计这些也就只能出现在面试题里面了。。。

兼容性

目前,主流浏览器(Chrome、Firefox、Safari 都已经在 release 版本支持了这个 APIFirefox 也已经在其 94 稳定版本支持了。

另外,Node 17 和 Deno 1.14 也已经实现了这个 API,未来一定会成为一个被广泛使用的 API 的,你可以放心大胆的用了。

参考

  • https://caniuse.com/?search=structuredClone
  • https://web.dev/structured-clone/



- EOF -

推荐阅读  点击标题可跳转

1、牛哇!居然在浏览器控制台安装 NPM 包!

2、postTask React的杀手锏被浏览器原生实现了?

3、json-server 30 秒无代码搭建一个完整的 REST API 服务


觉得本文对你有帮助?请分享给更多人

推荐关注「」,提升前端技能

点赞和在看就是最大的支持??

小羊羔锚文本外链网站长https://seo-links.cn 
回复列表
默认   热门   正序   倒序

回复:太卷了!浏览器也支持原生的深拷贝 API 了?

Powered by 小羊羔外链网 8.3.11

©2015 - 2024 小羊羔外链网

免费发软文外链 鄂ICP备16014738号-6

您的IP:216.244.66.232,2024-04-19 08:57:07,Processed in 0.05062 second(s).

支持原创软件,抵制盗版,共创美好明天!
头像

用户名:

粉丝数:

签名:

资料 关注 好友 消息