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

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

相关推荐
mapbar_front1 小时前
在职场生存中如何做个不好惹的人
前端
牧杉-惊蛰1 小时前
纯flex布局来写瀑布流
前端·javascript·css
一袋米扛几楼982 小时前
【软件安全】什么是XSS(Cross-Site Scripting,跨站脚本)?
前端·安全·xss
向上的车轮2 小时前
Actix Web适合什么类型的Web应用?可以部署 Java 或 .NET 的应用程序?
java·前端·rust·.net
XiaoYu20022 小时前
第1章 核心竞争力和职业规划
前端·面试·程序员
excel2 小时前
🧩 深入浅出讲解:analyzeScriptBindings —— Vue 如何分析 <script> 里的变量绑定
前端
蓝瑟2 小时前
AI时代程序员如何高效提问与开发工作?
前端·ai编程
林晓lx3 小时前
使用Git钩子+ husky + lint语法检查提高前端项目代码质量
前端·git·gitlab·源代码管理
王同学要变强3 小时前
【深入学习Vue丨第二篇】构建动态Web应用的基础
前端·vue.js·学习
沐怡旸3 小时前
【穿越Effective C++】条款14:在资源管理类中小心copying行为——RAII类的拷贝语义设计
c++·面试