深拷贝还在用lodash吗?来试试原装的structuredClone()吧!

一直以来深拷贝是前端的一个必考科目,面试来个手写deepClone函数你麻不麻?现在2025年了,得站起来了,来看看原装的structuredClone()函数吧。

以前的深拷贝

基本就两个实现方式

  • JSON.parse(JSON.stringify(obj))
  • lodash 的 cloneDeep

JSON.parse(JSON.stringify(obj))的优点是简单清晰,原生方法,缺点对象存在循环引用会报错:

js 复制代码
const obj = { a: 1 }
obj.b = obj

const obj1 = JSON.parse(JSON.stringify(obj))

还有Date对象会转化成字符串,Set、Map、正则、Error 对象,该方法会将其转成空对象字面量 { } ,如果存在 undefined,该方法会直接忽略

js 复制代码
const obj = { date: new Date('2025-07-10') }
const obj1 = JSON.parse(JSON.stringify(obj))
obj1.date // '2025-07-10T00:00:00.000Z'

const obj = {
    set: new Set([1, 2, 3]),
    map: new Map([['key1', 'value1'],['key2', 'value2']]),
    regex: /abc/gi,
    error: new Error('error!'),
    undefined: undefined
}
const obj1 = JSON.parse(JSON.stringify(obj))
console.log(obj1)

至于lodash老问题了

就算按需引入体积也不小

对有的体积有严格要求的项目直接就PASS了。

现在的深拷贝

废话不多说直接看代码

js 复制代码
const obj = { name: 'John' };
const clonedObj = structuredClone(obj);
console.log(obj === clonedObj); // false
obj.itself = obj
const clonedObj2 = structuredClone(obj);
console.log(clonedObj2.itself === obj); // false
console.log(clonedObj2.itself === clonedObj2); // true

const complex = {
    map: new Map([["key", "value"]]),
    set: new Set([1, 2, 3]),
    date: new Date(),
    regex: /abc/gi
};
const clone = structuredClone(complex);
console.log(clone); 
console.log(clone === complex); // false

优雅,简洁,舒服,还得是原装的。

StructuredClone()方法使用结构化克隆算法将给定的值进行深拷贝,该方法还支持把原值中的可转移对象转移(而不是拷贝)到新对象上。可转移对象与原始对象分离并附加到新对象,它们将无法在原始对象中被访问。这是MDN上的定义,转移这个定义有点理解不了,这是官方示例

js 复制代码
// 创建一个给定字节大小的 ArrayBuffer
const buffer1 = new ArrayBuffer(16);

const object1 = {
  buffer: buffer1,
};

// 克隆包含 buffer 的对象,并将其转移
const object2 = structuredClone(object1, { transfer: [buffer1] });

// 从克隆后的 buffer 创建数组
const int32View2 = new Int32Array(object2.buffer);
int32View2[0] = 42;
console.log(int32View2[0]);

// 从原 buffer 创建数组将抛出 TypeError
const int32View1 = new Int32Array(object1.buffer);

我粗浅的理解是键值之间的连接打断了,把缓存区值的指向指向了新的克隆对象,大概这么个意思,评论如果有大神知道,欢迎交流讨论!

最后看一下兼容性:

可以看到基本主流浏览器(除了IE)都支持了这个方法,Node17版本也支持了,可以放心大胆的用起来了。

相关推荐
Hello.Reader35 分钟前
Rust → WebAssembly 的性能剖析全指南
前端·rust·wasm
前端小巷子40 分钟前
Vue 2 Diff 算法
前端·vue.js·面试
奕辰杰5 小时前
关于npm前端项目编译时栈溢出 Maximum call stack size exceeded的处理方案
前端·npm·node.js
JiaLin_Denny6 小时前
如何在NPM上发布自己的React组件(包)
前端·react.js·npm·npm包·npm发布组件·npm发布包
_Kayo_7 小时前
VUE2 学习笔记14 nextTick、过渡与动画
javascript·笔记·学习
路光.7 小时前
触发事件,按钮loading状态,封装hooks
前端·typescript·vue3hooks
我爱996!7 小时前
SpringMVC——响应
java·服务器·前端
咔咔一顿操作8 小时前
Vue 3 入门教程7 - 状态管理工具 Pinia
前端·javascript·vue.js·vue3
kk爱闹8 小时前
用el-table实现的可编辑的动态表格组件
前端·vue.js