原来typescript也可以这么有趣

原来typescript也可以这么有趣,看好了,下面来看一段有趣的代码。

ts 复制代码
type Space = ' ' | '\n' | '\t';
type TrimLeft<S extends string> = S extends `${Space}${infer R}`
  ? TrimLeft<R>
  : S;

再看使用方式如下:

ts 复制代码
type TrimLeftStr = TrimLeft<'  Hello World  '>; // expected to be 'Hello World  '

查看结果如下图所示:

从来没有想过ts也可以去除字符串的空白,这简直令人难以想象。言归正传,我们先来解释一下以上代码的含义。

首先定一个了一个Space类型,这个类型是一个联合类型,第一个就是一段空白字符串,请记住中间是有空白的,第二个\n为换行符,表示回车换行,也就是将当前位置移到下一行的开头,第三个\t为水平制表符,也就是将当前位置移到下一个tab位置,总的说来,就是一个空白符的联合类型。

接着我们定义了一个去除左边空白的泛型,泛型S使用extends确定类型限定在字符串类型,因为只有字符串才需要清除空白,紧接着类型值我们使用一个三元表达式,判断条件就是泛型S是否继承一个模板字符串,恰恰有趣的就是这个模板字符串,这个模板字符串由我们定义的空白类型以及使用infer推断剩余字符串而组成的模板字符串。

note: infer是typescript的一个关键字,用于条件类型当中推断参数。

如果字符串满足模板字符串所定义的类型,那么就递归调用去除字符串类型,这时候参数变成了我们推断的R类型,否则就直接返回这个泛型S,这样也就达到了我们去除左边字符串空白的效果。

那么按照同样的思路,我们清除字符串右边空白也就可以轻而易举的实现,如下所示:

ts 复制代码
type Space = ' ' | '\n' | '\t';
type TrimRight<S extends string> = S extends `${infer R}${Space}`
  ? TrimRight<R>
  : S;

可以看到,我们只做了两个地方的改动,第一个当然就是类型名的更换,而另一个就是模板字符串中空白类型更换了一下位置。

如此一来,我们的trim类型,消除两边空白不就可以实现了吗,如下所示:

ts 复制代码
type Space = ' ' | '\n' | '\t';
type Trim<S extends string> = S extends `${Space}${infer R}${Space}`
  ? Trim<R>
  : S;

完美,typescript真的很有趣,这更像是js操作字符串了,不过话又说回来,这种操作恐怕只能自己玩玩,真要用到业务当中,嗯,我想应该不会有人写带空白字符串的类型吧。

以上代码段参考自type-challenges,并总结到我的代码段合集中。

相关推荐
小杍随笔6 小时前
【iNovel 前端架构深度解析:基于 Vue 3 + TypeScript + Tauri 的跨端小说写作工具】
前端·架构·typescript
sheeta19989 小时前
TypeScript 学习笔记
笔记·学习·typescript
晓杰'18 小时前
从0到1实现 Balatro 游戏后端(2):NestJS框架搭建与项目结构设计
后端·websocket·typescript·node.js·游戏开发·项目实战·nestjs
VillenK1 天前
版本依赖问题:vite-plugin-dts@3.1.0 与 jiti 的兼容性
前端·typescript·vite
轻口味1 天前
AI 时代全栈开发破局:TypeScript 生态实战,从入门到部署一站式通关
前端·mongodb·docker·ai·typescript·react·next.js
Czzzzlq1 天前
【无标题】
typescript·node.js·ai编程
mobº1 天前
Vue3 +TypeScript 项目总结
前端·javascript·typescript
用户056694948312 天前
🔥 我把 axios 接口封装,玩出了 NestJS 的感觉
typescript
索西引擎2 天前
【理论】TypeScript 函数重载:从 Vue 3 defineEmits 说起的类型安全实践
前端·typescript
漫游的渔夫2 天前
从 if-else 乱麻到状态机:前端开发者该怎么理解多 Agent 协作?
前端·人工智能·typescript