TypeScript类型体操--斐波那契序列

大家好,我是苏先生,2024年我的第一愿景就是做一个可行的副业,求大佬带我~~

好文推荐

前言

前边几篇文章我们一共实现了43 个工具类型,按照本专栏的规划,还差56个...

本节我们继续学习一个新的工具类型

提示

对于语法层面的知识点本系列(类型体操开头的标题)不会展开说明哈,可以自行搜索学习其他大佬的优质文章或者等我后续更新补充

题目

实现

首先,使用type关键字创建类型Fibonacci,它接收一个数值参数T

ts 复制代码
type Fibonacci<T extends number>

由于斐波那契序列一般指的都是非负整数索引,因此对于泛型T,需要对其取绝对值。至于负数,其本质上等价于先取正整数的斐波那契数然后再对其取反即可

如下,当为负数时,先通过Abs取正,再通过模版字符类型取反

ts 复制代码
type Fibonacci<T extends number> = 
  T extends Abs<T> 
  ? 
    _Fibonacci<T> 
  : 
    `-${_Fibonacci<Abs<T>>}`

不过,为了返回类型的一致性,将正整数T对应的斐波那契数也转换成字符串类型

ts 复制代码
type Fibonacci<T extends number> = 
  T extends Abs<T> 
  ? 
    `${_Fibonacci<T>}` 
  : 
    `-${_Fibonacci<Abs<T>>}`

另外

已知斐波那契序列的性质如下

text 复制代码
f(n)=f(n-2)+f(n-1)

我们搞一个js函数来观察下

当n为0时其结果为0,当n为1或2的时候,其结果为1,这些结果是恒定的,可以将其前置处理掉

ts 复制代码
type _Fibonacci<T extends number> = 
  T extends 0 
  ? 
    0 
  : 
    T extends 1 | 2 
    ? 
      1
    :
      ...

省略号部分即T>=3的时候,由截图可知,它对应的斐波那契数为

text 复制代码
f(3)=f(1)+f(2)

则,取parent的首字母P为f(2),即f(n-1),grandfather的首字母G为f(1),即f(n-2),声明变量如下

ts 复制代码
type _Fibonacci<
  T extends number, 
  P extends number[] = [1],
  G extends number[] = [1]
> = ...

则对于大于等于3的数,只需要对其执行递归

ts 复制代码
_Fibonacci<T,f(n-1),f(n-2)>

然后取P和G的length和就是最终的答案了

ts 复制代码
[...P,...G]['length']

那现在有两个问题需要解决

1.递归何时结束?

声明变量I表示当前正在计算的斐波那契数

ts 复制代码
type _Fibonacci<
  ..., 
  I extends number[] = [1,1,1], 
  ...> = ...

每递归一次,就将I的length加1

ts 复制代码
_Fibonacci<...,[...I,1],...>

当I['length']与T相等时,递归结束

ts 复制代码
type _Fibonacci<
  ..., 
  I extends number[] = [1,1,1], 
  ...> 
  = 
    ...
    I['length'] extends T 
    ? 
      '出口' 
    : 
      '递归'

2.传入递归的f(n-1)和f(n-2)是什么?

结合定义和前文关于出口的说明,只需要依次向前推进一位即可

当T=6时

进入递归,P为5即f(n-1),G为4即f(n-2)

ts 复制代码
_Fibonacci<T,f(6),f(5)>

当T=7时

进入递归,P为6即f(n-1),G为5即f(n-2)

ts 复制代码
_Fibonacci<T,f(7),f(6)>

故,完整实现如下

ts 复制代码
type Abs<
    N extends number
> = 
`${N}` 
  extends `-${infer R extends number}` 
? R 
: N;

type _Fibonacci<
  T extends number, 
  I extends number[] = [1,1,1], 
  P extends number[] = [1],
  G extends number[] = [1]
> = 
T extends 0 
? 
    0 
: 
    T extends 1 | 2 
    ? 
        1
    : 
        T extends I['length'] 
        ? 
            [...P,...G]['length']
        :
            _Fibonacci<T,[...I,1],[...P,...G],P> 

type Fibonacci<T extends number> = 
  T extends Abs<T> 
  ? 
    `${_Fibonacci<T>}` 
  : 
    `-${_Fibonacci<Abs<T>>}`

使用如下

下期预告

  • 【知识点】巧用装饰器自动修正TypeScript中的类this指向

-【类型体操】去除数组指定元素

接收数组类型的 T 和数字或数组类型的 U 为参数,会返回一个去除 U 中元素的数组 T

ts 复制代码
type Res = Without<[1, 2], 1>; // [2]
type Res1 = Without<[1, 2, 4, 1, 5], [1, 2]>; // [4, 5]
type Res2 = Without<[2, 3, 2, 3, 2, 3, 2, 3], [2, 3]>; // []

如果本文对您有用,希望能得到您的点赞和收藏

订阅专栏 ,每更新1-2篇类型体操,等你哟😎

相关推荐
Hao_Harrision14 小时前
50天50个小项目 (React19 + Tailwindcss V4) ✨| AnimatedCountdown(倒计时组件)
前端·typescript·react·tailwindcss·vite7
ttod_qzstudio15 小时前
从 Switch-Case 到自注册工厂:优雅的驱动行为管理系统重构
重构·typescript·工厂模式
CodeCaptain1 天前
CocosCreator3.8.x 解析Tiled1.4.x【瓦片图层、对象图层、图像图层、组图层】的核心原理
经验分享·游戏·typescript·cocos2d
树叶会结冰1 天前
TypeScript---对象:不自在但实在
前端·javascript·typescript
江公望1 天前
tsconfig配置文件增加“esModuleInterop“: true,有什么作用?
typescript
wordbaby1 天前
三个核心概念,帮你彻底打通 TypeScript 泛型
前端·typescript
guangzan2 天前
Zod:TypeScript 类型守卫与数据验证
ai·typescript·zod
军军君012 天前
Three.js基础功能学习四:摄像机与阴影
开发语言·前端·javascript·3d·typescript·three·三维
EndingCoder3 天前
安装和设置 TypeScript 开发环境
前端·javascript·typescript
一个处女座的程序猿O(∩_∩)O3 天前
现代前端开发的三大支柱:TypeScript、ESLint、Prettier 深度解析与完美协作
javascript·typescript