前端深度拷贝以及使用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)
}

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

相关推荐
代码猎人1 分钟前
substring和substr有什么区别
前端
pimkle1 分钟前
visactor vTable 在移动端支持 ellipsis 气泡
前端
donecoding1 分钟前
告别 scrollIntoView 的“越级滚动”:一行代码解决横向滚动问题
前端·javascript
0__O1 分钟前
如何在 monaco 中实现自定义语言的高亮
前端·javascript·编程语言
Jasmine_llq3 分钟前
《P3200 [HNOI2009] 有趣的数列》
java·前端·算法·线性筛法(欧拉筛)·快速幂算法(二进制幂)·勒让德定理(质因子次数统计)·组合数的质因子分解取模法
呆头鸭L4 分钟前
快速上手Electron
前端·javascript·electron
Aliex_git9 分钟前
性能指标笔记
前端·笔记·性能优化
秋天的一阵风9 分钟前
🌟 藏在 Vue3 源码里的 “二进制艺术”:位运算如何让代码又快又省内存?
前端·vue.js·面试
松涛和鸣9 分钟前
48、MQTT 3.1.1
linux·前端·网络·数据库·tcp/ip·html
helloworld也报错?10 分钟前
保存网页为PDF
前端·javascript·pdf