You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
俗话说:“一入 any 深似海,从此类型是路人”,any 不会做任何类型检查,下面这段代码运行之后肯定会报 TypeError: value.toLocaleLowerCase is not a function 错误,并且这种错误只能在运行时才会发现,应尽可能的避免使用 any,否则就失去了使用 TypeScript 的意义。
Exclude 的第一个类型 T 是一个联合类型,T extends U ? never : T 这块的判断是,如果类型匹配,返回 never 就什么也没有,不匹配则返回。我们的例子中,age 类型匹配没有返回,返回的是未匹配的 name。
再通过 Pick 取出 Exclude 的结果,这样也就相当于达到了过滤的效果。
/** * Exclude from T those types that are assignable to U */typeExclude<T,U>=TextendsU ? never : T;typeOmit<T,Kextendskeyofany>=Pick<T,Exclude<keyofT,K>>;// 下面两个是等价的constuser: Omit<Person,'name'|'age'>={}// 等价于constuser: Pick<Person,Exclude<'name'|'age','age'>>={name: '五月君'}
增强的面向对象能力
ES6 时代为 JavaScript 增加了 class “语法糖”,可以方便的去定义一个类并通过 extends 关键词实现继承,尽管 ES6 中的 class 本质上仍是基于原型链实现的,但代码编写方式看起来简洁多了(以 class 关键词进行的面向对象编程)。
和其它的面向对象编程语言相比较,会发现 JavaScript 中的 class 少了好多功能。一个常见需求是不能私有化类成员,为了达到这个目的,通常有几种做法:在属性或方法前加上 _ 表示私有化,这属于命名规则约束、使用 symbol 的唯一性实现私有化。
TypeScript 是 JavaScript 的超集,JavaScript 能够做的事情,它都可以做且还增加了很多功能,例如静态类型、增强的面向对象编程能力等。
本文是笔者日常学习、使用 TypeScript 过程中自己记录的一些知识点,现在总结分享给大家。包含了做为初学者在学习 TypeScript 时应关注的核心知识,在掌握了这些知识点后,就是在项目中的灵活应用,文末推荐了一个基于 TypeScript 的前后端项目,做为学习可参考(文末阅读原文查看)。
下图为本文概览:
社区发展现状调研
近些年 TypeScript 在前端圈备受推崇,正在被越来越多的前端开发者所接受,包括在 Node.js 后端开发中同样如此。
下面从几份开发者报告调查数据看 TypeScript 的发展现状。
2022 年前端开发者现状报告
本报告来源于 https://tsh.io/state-of-frontend/#report 调查面比较广泛。来自 125 个国家的 3703+ 前端开发者及 19 位前端专家参与了该次调查。
关于 TypeScript ,过去一年里有将近 84.1% 的参与者表示使用过。
对于 TypeScript 前景描述,近 43% 开发者表示 “TypeScript 将超越 JavaScript 成为新的前端标准” 还是比较看好的。
2021 年 Node.js 年度报告
下面来看一份来国内的 Node.js 2021 年度开发者调查报告 https://nodersurvey.github.io/reporters。
从 “代码转译” 角度调研,有 47.5% 的开发者使用了 TypeScript。在 Node.js 后端框架中 Nest.js 最近几年也是一个热门选择,该框架一个特点是完全基于 TypeScript,受到了更多开发者的青睐。
一些其它的调查者报告数据
还有一些其它的调查报告数据,例如 2021 年 Stack Overflow 调查者报告、Google 搜索量趋势,NPM 下载量趋势 从上面可以看出 TypeScript 近几年的发展趋势还是很快的。
现在的你可能还在考虑要不要学习 TypeScript,但在将来也许会是每个前端开发者、Node.js 开发者必备的技能。
TS 困扰与收益
初学者的 TS 困扰
做为一个初学者刚开始使用 TypeScript 时,你无法在使用一些 JavaScript 的思维来编码,下面的一部分观点也许是每一个 TypeScript 初学者都会遇到的疑问。
TS 的收益
开发环境搭建
TypeScript 不能被浏览器或 Node.js 和 Deno 这些运行时所理解,最终要编译为 JavaScript 执行,我们需要一个 compiler 做编译。另外你可能会想到在 Deno 中不是可以直接写 TypeScript 吗,本质上 Deno 也不是直接运行的 TypeScript,同样需要先编译为 JavaScript 来运行。
在线编译
想尝试一下而不想本地安装的,可以看看以下这两个在线工具:
tsc VS ts-node
tsc 是将 TypeScript 代码编译为 JavaScript 代码,全局安装
npm install -g typescript
即可得到一个 tsc 命令,之后通过tsc hello.ts
编译 typescript 文件。与 tsc 不同的是 ts-node 是编译 + 执行。可以在开发时使用 ts-node,生产环境使用 tsc 编译。
框架/库支持
使用 CRA、Vite 这种库创建出来的前端 TS 项目、及后端 Nest.js 这样的框架,默认都是支持 TypeScript 的,一些基础的 tsconfig.json 配置文件,也都帮你配置好了。
初始化配置文件
TypeScript 编译时会使用
tsconfig.json
文件做为配置文件,该配置所在的项目会被认为是根目录。使用
npx tsc --init
命令快速创建一个 tsconfig.json 配置文件,具体的配置可参考 文档。数据类型核心概念
TypeScript 除了包含 JavaScript 已有的
string
、number
、boolean
、symbol
、bigint
、undefined
、null
、array
、object
数据类型之外,还包括tuple
、enum
、any
、unknown
、never
、void
、范型
概念及类型声明符号ineterface
、type
。基础数据类型
在参数名称后面使用冒号指定参数类型,同时也可在类型后面赋默认值,const 声明的变量必须要赋予默认值否则 IDE 编译器会提示错误,let 则不是必须的。
对于一些基础的数据类型,如果后面有值,TS 可以自动进行类型推断,不需要显示声明,例如
const nickname: string = '五月君'
等价于const nickname = '五月君'
。数组 VS 元组
数组(Array)通常用来表示所有元素类型相同的集合,也可以使用数组泛型:
Array<element type>
允许这个集合中存在多种类型。元组(Tuple)允许一个已知元素数量的数组中各元素的类型可以是不同的,通常是事先定义好的不能被修改,元组的定义和赋值必须要一一对应。
元组更严格一些,不可出现越界操作,例如 list1 只有三个元素,下标从 0 ~ 2,如果执行 list1[3] 就会报错,如下所示,这在数组操作中是不会出现报错的。
函数类型声明
可选参数使用 “?” 符号声明,可选参数、函数参数的默认值需要声明在必选参数之后。函数也可以声明返回值类型,在某些情况下不需要显示声明,能够自动推断出返回值类型。
当一个函数没有返回值时用 void 表示,在 JavaScript 中一个函数没有返回值,它的结果也等同于 undefined。
任何值 - any? 还是 unknown?
俗话说:“一入 any 深似海,从此类型是路人”,any 不会做任何类型检查,下面这段代码运行之后肯定会报
TypeError: value.toLocaleLowerCase is not a function
错误,并且这种错误只能在运行时才会发现,应尽可能的避免使用 any,否则就失去了使用 TypeScript 的意义。unknown 也表示任何值,相比于 any 它更加严谨,在使用上会有很多限制。
下面代码在编译时
fn1
函数会报错TSError: ⨯ Unable to compile TypeScript: Object is of type 'unknown'
。不对 unknown 声明的类型做任何取值操作,是没问题的,正如上例的
fn2
函数。这个时候就有疑问了,既然什么都不能操作,有什么应用场景呢?unknown 与类型守卫
unknown 的意义在于我们可以结合 “类型守卫” 在声明的函数或其它块级作用域内获取精确的参数类型。这样在运行时也能避免出现类型错误这种常见问题。
例如,想对一个数组执行
length
操作时,首先通过Array.isArray
进一步精确参数的类型之后,在做一些操作。使用 is 关键词自定义类型守卫
{参数名称} is {参数类型}
,这个意思是告诉 TypeScript 函数 isString() 返回的是一个 string 类型。例如,react-query 这个库返回的 error 默认为 unknown 类型,如果在 render 时直接这样写
<p>${error.message}<p>
是不行的,一个解决方案是拿到错误后用类型守卫处理,如下所示:另外一种方案是在调用它的 Api 时直接传入一个 Error 对象:
useMutation<TResponse, Error, string>()
枚举
枚举定义了一组相关值的集合,通过描述性的常量声明使代码更具可读性。默认情况下枚举将字符串的值存储为从 0 开始的数字,也可以显示声明为字符串。
交叉、联合类型、类型别名
交叉类型:是将多个类型合并为一个类型,使用符号 & 表示。例如,将 TPerson & TWorker 合并为一个新的类型 User。
注意,声明类型时不要与系统的关键词冲突,例如上例中的 Worker 尽管使用 interface 声明时没有提示报错,但使用时会有提示,因此才改为 TWorker。
联合类型:表示一个变量通常由多个类型组成,这之间是一种或的关系,使用
|
符号声明。意思是 id 即可以是 string 也可以是 number 类型。类型别名:给一个类型起一个新名字。如果另外一个字段和 id 一样也是由相同的多个类型组成,就要用到类型别名了,使用
type
关键字将多个基本类型声明为一个自定义的类型,这种是interface
替代不了的。class、interface、type 类型声明
TypeScript 中使用 class、interface、type 关键词均可声明类型。class 声明的是一个类对象,这个好理解,容易迷惑的地方在于 interface 和 type 两者分别该用于何处。
interface 和 type 非常相似,大多数情况下 interface 的特性都可以使用 type 实现。但两者还是有些区别:
从概念上每个人都有不同的理解,对于团队来说无论使用哪一个,都应该保持好统一的规范。在 TypeScript 官网文档中也提到了 如果不清楚该使用哪一个,请使用 interface 直到您需要 type。
鸭子类型
鸭子类型在程序设计中是动态类型的一种风格,它的关注点在于对象的行为能做什么,而不是关注对象所属的类型。这个概念的名字源自一个 “鸭子测试”,可以解释为 :
下例,使用 interface 定义了两个类型 Duck、Bird,它们都有共同的特征 “走路”,于是对两个类型分别定义了 walk() 方法。test() 方法中期望的 value 类型为 Duck,但实际调用时传的 value 是 Bird,在 TypeScript 中是可以编译通过的。
TypeScript 的鸭子类型是面向接口编程,虽然我们的例子 Duck、Bird 类型不一样,但是都有共同的接口,是可以编译通过的,对于面向对象编程的语言,例如 Java 就不行了。
范型
在参数和返回类型未知的情况下,可以使用范型来传递类型,保证组件内的类型安全。
一个有用的场景是用范型定义接口返回对象。例如,接口返回值 Result 对象,每一个接口返回的 data 是不一样的,这时 Result 对象的 data 属性就适合用范型作为类型进行传递。
在函数内部使用范型变量时,存在类型约束,不能随意操作它的属性。例如下例,需要事先定义范型 T 拥有 length 属性。
范型定义也可以有多个。例如,swap 函数交换两个变量。
实用类型工具
Utility type(实用类型)
不是一种新的类型,基于类型别名实现的一个工具集的自定义类型。包括:Parameters<T>
、Omit<T,K>
。Parameters
Parameters<T>
:接收一个范型 T,这个 T 是一个 Function,将会提取这个函数的返回值为tuple
。例如,两个函数 fn2 的参数同 fn1 相等,这时使用 Parameters 就很合适。
Pritial
Pritial<T>
:将范型 T 的所有属性变为可选。例如,user1 我必须写 name、age,而 user2 就不用了,现在都变为选填了,只填写需要的数据。实现原理:实用 keyof 关键词取出 T 的所有的属性,遍历过程为每个属性加了一个
?
符号,也就表示可选的。Pick
Pick<T,K>
:选取范型 T 中指定的部分属性,如果需要选出多个属性,用联合类型指定。实现原理:首先校验第二个类型 K 的属性必须在第一个类型 T 里面,之后遍历传入的联合类型 K,形成一个新的类型。
Omit
Omit<T,K>
:与 Pick 相反,将第一个范型 T 中的部分属性删除,如果要删除多个属性,用联合类型指定需要删除的属性。实现原理:Omit 使用了 Pick 和 Exclude 组合来实现的,这个地方有点绕:
T extends U ? never : T
这块的判断是,如果类型匹配,返回 never 就什么也没有,不匹配则返回。我们的例子中,age 类型匹配没有返回,返回的是未匹配的 name。增强的面向对象能力
ES6 时代为 JavaScript 增加了 class “语法糖”,可以方便的去定义一个类并通过 extends 关键词实现继承,尽管 ES6 中的 class 本质上仍是基于原型链实现的,但代码编写方式看起来简洁多了(以 class 关键词进行的面向对象编程)。
和其它的面向对象编程语言相比较,会发现 JavaScript 中的 class 少了好多功能。一个常见需求是不能私有化类成员,为了达到这个目的,通常有几种做法:在属性或方法前加上
_
表示私有化,这属于命名规则约束、使用 symbol 的唯一性实现私有化。TypeScript 中增强了面向对象的编程能力,具备类的访问权限控制、接口、模块、类型注解等功能。
类成员访问权限控制
对象的成员属性或方法如果没有被封装,实例化后在外部就可通过引用获取,对于用户 phone 这种数据,是不能随意被别人获取的。
封装性做为面向对象编程重要特性之一,它是把类内部成员属性、成员方法统一保护起来,只保留有限的接口与外部进行联系,尽可能屏蔽对象的内部细节,防止外部随意修改内部数据,保证数据的安全性。
同传统的面向对象编程语言类似,TypeScript 提供了 3 个关键词 public、private、protected 控制类成员的访问权限。
接口
接口是一种特殊的抽象类,与抽象类不同的是,接口没有具体的实现,只有定义,通过 interface 关键词声明。
TypeScript 对接口的定义是这样的:
TypeScript 中只能单继承,但可以实现多个接口。
面向对象程序设计概念不止这些,参见这篇文章 #41。
总结
用还是不用?
最终要不要用 TypeScript,还要结合项目规模、维护周期、团队成员多方面看,以下为个人的一些理解:
该怎么学习?
文章开头我们看了一些 TypeScript 社区发展现状调研报告,从目前使用情况、发展趋势看,已然成为前端开发者的必备技能之一。如果你还在犹豫要不要学习 TypeScript,那我建议你在时间允许的情况,开始做一些尝试吧。
下面从个人角度,总结一些建议:
The text was updated successfully, but these errors were encountered: