typescript类型体操--驼峰下划线互转

需求

1.对于给定下划线形式的字符串类型T,实现一个泛型CamelCase,返回驼峰形式的类型G。

2.对于给定驼峰形式的字符串类型G,实现一个泛型UnderlineCase,返回下划线形式的类型T。

typescript 复制代码
type camelCaseResult = CamelCase<'hello_my_love'> // expected to 'helloMyLove'
typescript 复制代码
type UnderlineCaseResult = 
UnderlineCase<'helloMyLove'> // expected to 'hello_my_love'

前置知识

分析

CamelCase

对于下划线转小驼峰,我们实际需要转换的字符串中可能的单词个数(以_分隔)是不定的,为此,我们该操作会涉及到递归,递归终止条件为剩余子串是否含有_

使用extends做类型约束时,我们可以结合ts类型推断中使用模版字符串加infer关键字指定局部变量。

在我的观察中,推断时,除非指定非字母的分隔符,否则推断中每次匹配都将把字符串按单个字母拆分,以下就是一个例子:

typescript 复制代码
type ITest<Str extends string> = Str extends `${infer A}_${infer B}${infer C}`
    ? `A-(${A})/B-(${B})/C-(${C})` : Str

于是我们extends约束的类型为

typescript 复制代码
`${infer First}_${infer Second}${infer Rest}`

以'hello_my_love'为例,首次匹配时:

typescript 复制代码
First -- 'hello';
Second -- 'm'
Rest -- "y_love"

显而易见地,hello不需要处理,m需要转为My_love则进入到下一次匹配中(递归操作)。

第二次匹配时:

typescript 复制代码
First -- 'y';
Second -- 'l'
Rest -- "ove"

因此,CamelCase这个泛型为:

typescript 复制代码
type CamelCase<Str extends string> =
    Str extends `${infer First}_${infer Second}${infer Rest}`
        ? `${First}${Uppercase<Second>}${CamelCase<Rest>}`
        : Str

UnderlineCase

易知,关键是识别出helloMyLove中的ML,但ts的类型推断中似乎并没有已存在的类型判断某个字母是否是大写字母,只好手写了:

typescript 复制代码
type UpperLetter = 'A'|'B'|'C'
    |'D'|'E'|'F'|'G'|'H'|'I'|'J'|'K'|'L'|'M'|'N'|'O'|'P'
    |'Q'|'R'|'S'|'T'|'U'|'V'|'W'|'X'|'Y'|'Z'

按照Camelcase中的思路,约束类型可以为:

typescript 复制代码
`${infer First}${infer Upper}${infer Rest}`

第一次匹配时:

typescript 复制代码
First -- 'h'
Upper -- 'e'
Rest -- 'lloMyLove'

第二次匹配时:

typescript 复制代码
First -- 'l'
Upper -- 'l'
Rest -- 'oMyLove'

第三次匹配时:

typescript 复制代码
First -- 'o'
Upper -- 'M'
Rest -- 'yLove'

...

我们应该做的应该是把大写字母ML分别转为_m_l,因此,新建一个类型UnderlineChar实现:

typescript 复制代码
type UnderlineChar<Str extends string> =
    Str extends UpperLetter
        ? `_${Lowercase<Str>}`
        : Str

最终的结果为:

typescript 复制代码
type UnderlineCase<Str extends string> =
    Str extends `${infer First}${infer Upper}${infer Rest}`
        ? `${UnderlineChar<First>}${UnderlineChar<Upper>}${UnderlineCase<Rest>}`
        : Str

应用场景

CamelCase

typescript 复制代码
function toCamelString<T extends string,>(str:T):CamelCase<T>
function toCamelString(str){
    return str.replace(/_(\w)/g, function(_, letter){
        return letter.toUpperCase();
    });
}

UnderlineCase

typescript 复制代码
function toUnderlineString<T extends string>(str:T):UnderlineCase<T>
​
function toUnderlineString(str:string){
    return str.replace(/([A-Z])/g,function (_,letter){
        return '_'+letter.toLowerCase()
    })
}
相关推荐
meilindehuzi_a19 小时前
构建基于 RESTful 架构的 TodoList 全栈应用:从前后端理论到 TypeScript/Bun 实战
架构·typescript·restful
云水一下20 小时前
Vue.js从零到精通系列(七):高级特性实战——Teleport、异步组件、自定义指令与TypeScript深度结合
前端·vue.js·typescript
濮水大叔21 小时前
浅论CabloyJS全栈框架提供的“两级页签”机制
typescript·node.js·next.js
Asize21 小时前
Bun + TypeScript:AI 时代的后端开发入门
人工智能·typescript·bun
退休倒计时1 天前
【每日一题】LeetCode 53. 最大子数组和 TypeScript
数据结构·算法·leetcode·typescript
小林ixn2 天前
你以为你懂 + 号?看完这篇 Bun + TS 实战,才发现以前全写错了
前端·javascript·typescript
晓杰'2 天前
从0到1实现Balatro游戏后端(8):Skip Blind与Tag奖励机制设计与实现
后端·websocket·typescript·项目实战·nestjs·状态管理·游戏服务器
YHHLAI2 天前
从零搭建一个 RESTful Todo 服务 —— Bun + TypeScript 全栈最小闭环
后端·typescript·restful
退休倒计时2 天前
【每日一题】LeetCode 19. 删除链表的倒数第 N 个结点 TypeScript
leetcode·链表·typescript
Sammyyyyy2 天前
月之暗面 Kimi Code 0.4.0 发布,终端 AI 编码助手全面采用 TypeScript,实现毫秒级启动
前端·javascript·人工智能·ai·typescript·servbay