前端深度拷贝以及使用TS标注类型

前言

在前端中对象深拷贝是比较常见的面试题,在日常开发使用中也很常见,很多人在使用的时候也许会有一些类似于JSON.stringify这种技巧来实现深度克隆,但是在写一些中大型的项目中都会实现deepClone的工具方法, 今天介绍一下我在项目中如何实现以及如何去标注deepClone函数类型的

deepClone

实现思路

  1. 判断传参值类型,当为null或者不为object时, 直接返回传参
  2. 为了避免对象的循环引用,一般使用 WeakMap 来储存每次循环时的健值,同时 WeakMap 也不会影响JS的垃圾回收。
  3. 每次判断一下传入的obj是否在cache中已经存在,如果存在就将值直接返回
  4. 此时已经确认obj是一个对象,继续确认它的类型,是否为数组
  5. 到这一步已经确认了传入的obj是对象且并未被缓存过,缓存该对象
  6. 递归调用自身,直到整个对象被拷贝完成

实现JS代码

先上一个JS版本的代码,关于如何标注类型后续完成

scss 复制代码
function deepClone(obj) {
    var cache = new WeakMap();
    function _deepClone(obj) {
        if (obj === null || typeof obj !== 'object') {
            return obj;
        }
        if (cache.has(obj)) {
            return cache.get(obj);
        }
        var result = Array.isArray(obj)
            ? []
            : {};
        cache.set(obj, result);
        for (var key in obj) {
            if (obj.hasOwnProperty(key)) {
                result[key] = _deepClone(obj[key]);
            }
        }
        return result;
    }
    return _deepClone(obj);
}

TS类型标注

看到很多人在TS中就参数使用一个any作为标注,这样的话在调用时以及调用之后类型不明确,既然是一个对象深度克隆函数, 那么返回的肯定是和传入参数是同一个类型,这里可以使用一个泛型U来制定类型, 返回值也为该类型,在内部函数泛型T的循环中,result每次保存的类型都是泛型T的某个属性或者对象, 所以result可以看作是Partial<T>, 在本次对象的所有循环赋值结束后 result 的类型才变为 T, 所以可以使用as断言result的类型

vbnet 复制代码
function deepClone<U>(obj: U): U {
  const cache = new WeakMap()
  function _deepClone<T>(obj: T): T {
    if (obj === null || typeof obj !== 'object') {
      return obj
    }
    if (cache.has(obj)) {
      return cache.get(obj)
    }
    const result = Array.isArray(obj)
      ? ([] as unknown as Partial<T>)
      : ({} as Partial<T>)
    cache.set(obj, result)

    for (const key in obj) {
      if ((obj as Object).hasOwnProperty(key)) {
        result[key as keyof T] = _deepClone(obj[key])
      }
    }
    return result as T
  }
  return _deepClone(obj)
}

这样的类型就没问题了。使用函数后赋值类型也能明确了。

相关推荐
半桔20 分钟前
【Linux手册】从接口到管理:Linux文件系统的核心操作指南
android·java·linux·开发语言·面试·系统架构
开开心心就好31 分钟前
电脑息屏工具,一键黑屏超方便
开发语言·javascript·电脑·scala·erlang·perl
江号软件分享32 分钟前
有效保障隐私,如何安全地擦除电脑上的敏感数据
前端
web守墓人2 小时前
【前端】ikun-markdown: 纯js实现markdown到富文本html的转换库
前端·javascript·html
Savior`L2 小时前
CSS知识复习5
前端·css
许白掰2 小时前
Linux入门篇学习——Linux 工具之 make 工具和 makefile 文件
linux·运维·服务器·前端·学习·编辑器
中微子6 小时前
🔥 React Context 面试必考!从源码到实战的完整攻略 | 99%的人都不知道的性能陷阱
前端·react.js
秋田君7 小时前
深入理解JavaScript设计模式之命令模式
javascript·设计模式·命令模式
中微子7 小时前
React 状态管理 源码深度解析
前端·react.js
风吹落叶花飘荡8 小时前
2025 Next.js项目提前编译并在服务器
服务器·开发语言·javascript