- {
- let str: string = ‘this is string’;
- let num: number = 1;
- let bool: boolean = true;
- }
- {
- const str: string = ‘this is string’;
- const num: number = 1;
- const bool: boolean = true;
- }
- {
- let str = ‘this is string’; // 等价
- let num = 1; // 等价
- let bool = true; // 等价
- }
- {
- const str = ‘this is string’; // 不等价
- const num = 1; // 不等价
- const bool = true; // 不等价
- }
类型推断
在 TypeScript 中,类型标注声明是在变量之后(即类型后置),它不像 Java 语言一样,先声明变量的类型,再声明变量的名称。 使用类型标注后置的好处是编译器可以通过代码所在的上下文推导其对应的类型,无须再声明变量类型,具体示例如下:- {
- let x1 = 42; // 推断出 x1 的类型是 number
- let x2: number = x1; // ok
- }
- {
- /* 根据参数的类型,推断出返回值的类型也是 number /
- functionadd1(a: number, b: number) {
- return a + b;
- }
- const x1= add1(1, 1); // 推断出 x1 的类型也是 number
- /* 推断参数 b 的类型是数字或者 undefined,返回值的类型也是数字 /
- functionadd2(a: number, b = 1) {
- return a + b;
- }
- const x2 = add2(1);
- const x3 = add2(1, ‘1’); // ts(2345) Argument of type ‘“1”‘ is not assignable to parameter of type ‘number | undefined
- }
上下文推断
通过类型推断的例子,我们发现变量的类型可以通过被赋值的值进行推断。除此之外,在某些特定的情况下,我们也可以通过变量所在的上下文环境推断变量的类型,具体示例如下:- {
- type Adder = (a: number, b: number) =>number;
- const add: Adder = (a, b) => {
- return a + b;
- }
- const x1 = add(1, 1); // 推断出 x1 类型是 number
- const x2 = add(1, ‘1’); // ts(2345) Argument of type ‘“1”‘ is not assignable to parameter of type ‘number
- }
- {
- let str = ‘this is string’; // str: string
- let num = 1; // num: number
- let bool = true; // bool: boolean
- }
- {
- const str = ‘this is string’; // str: ‘this is string’
- const num = 1; // num: 1
- const bool = true; // bool: true
- }
字面量类型
在 TypeScript 中,字面量不仅可以表示值,还可以表示类型,即所谓的字面量类型。 目前,TypeScript 支持 3 种字面量类型:字符串字面量类型、数字字面量类型、布尔字面量类型,对应的字符串字面量、数字字面量、布尔字面量分别拥有与其值一样的字面量类型,具体示例如下:- {
- let specifiedStr: ‘this is string’ = ‘this is string’;
- let specifiedNum: 1 = 1;
- let specifiedBoolean: true = true;
- }
字面量类型是集合类型的子类型,它是集合类型的一种更具体的表达。比如 ‘this is string’ (这里表示一个字符串字面量类型)类型是 string 类型(确切地说是 string 类型的子类型),而 string 类型不一定是 ‘this is string’(这里表示一个字符串字面量类型)类型,如下具体示例:
- {
- let specifiedStr: ‘this is string’ = ‘this is string’;
- let str: string = ‘any string’;
- specifiedStr = str; // ts(2322) 类型 ‘“string”‘ 不能赋值给类型 ‘this is string’
- str = specifiedStr; // ok
- }
字符串字面量类型
一般来说,我们可以使用一个字符串字面量类型作为变量的类型,如下代码所示:- let hello: ‘hello’ = ‘hello’;
- hello = ‘hi’; // ts(2322) Type ‘“hi”‘ is not assignable to type ‘“hello”‘
- type Direction = ‘up’ | ‘down’;
- functionmove(dir: Direction) {
- // …
- }
- move(‘up’); // ok
- move(‘right’); // ts(2345) Argument of type ‘“right”‘ is not assignable to parameter of type ‘Direction’
通过使用字面量类型组合的联合类型,我们可以限制函数的参数为指定的字面量类型集合,然后编译器会检查参数是否是指定的字面量类型集合里的成员。
因此,相较于使用 string 类型,使用字面量类型(组合的联合类型)可以将函数的参数限定为更具体的类型。这不仅提升了程序的可读性,还保证了函数的参数类型,可谓一举两得。数字字面量类型及布尔字面量类型
数字字面量类型和布尔字面量类型的使用与字符串字面量类型的使用类似,我们可以使用字面量组合的联合类型将函数的参数限定为更具体的类型,比如声明如下所示的一个类型 Config:- interface Config {
- size: ‘small’ | ‘big’;
- isEnable: true | false;
- margin: 0 | 2 | 4;
- }
介绍完三种字面量类型后,我们再来看看通过 let 和 const 定义的变量的值相同,而变量类型不一致的具体原因。
我们先来看一个 const 示例,如下代码所示:- {
- const str = ‘this is string’; // str: ‘this is string’
- const num = 1; // num: 1
- const bool = true; // bool: true
- }
- {
- let str = ‘this is string’; // str: string
- let num = 1; // num: number
- let bool = true; // bool: boolean
- }
- str = ‘any string’;
- num = 2;
- bool = false;
Literal Widening
所有通过 let 或 var 定义的变量、函数的形参、对象的非只读属性,如果满足**指定了初始值且未显式添加类型注解**的条件,那么它们推断出来的类型就是指定的初始值字面量类型拓宽后的类型,这就是字面量类型拓宽。
下面我们通过字符串字面量的示例来理解一下字面量类型拓宽:- {
- let str = ‘this is string’; // 类型是 string
- let strFun = (str = ‘this is string‘) => str; // 类型是 (str?: string) => string;
- const specifiedStr = ‘this is string’; // 类型是 ‘this is string’
- let str2 = specifiedStr; // 类型是 ‘string’
- let strFun2 = (str = specifiedStr) => str; // 类型是 (str?: string) => string;
- }
- {
- const specifiedStr: ‘this is string’ = ‘this is string’; // 类型是 ‘“this is string”‘
- let str2 = specifiedStr; // 即便使用 let 定义,类型是 ‘this is string’
- }
Type Widening
比如对 null 和 undefined 的类型进行拓宽,通过 let、var 定义的变量如果满足未显式声明类型注解且被赋予了 null 或 undefined 值,则推断出这些变量的类型是 any:- {
- let x = null; // 类型拓宽成 any
- let y = undefined; // 类型拓宽成 any
- /* ——-分界线———- /
- const z = null; // 类型是 null
- /* ——-分界线———- /
- let anyFun = (param = null) => param; // 形参类型是 null
- let z2 = z; // 类型是 null
- let x2 = x; // 类型是 null
- let y2 = y; // 类型是 undefined
- }
Type Narrowing
在 TypeScript 中,我们可以通过某些操作将变量的类型由一个较为宽泛的集合缩小到相对较小、较明确的集合,这就是 “Type Narrowing”。 ts中的类型守卫就是条件判断,在 TypeScript 类型层面,条件判断顺带的会触发类型缩小 比如,我们可以使用类型守卫将函数参数的类型从 any 缩小到明确的类型,具体示例如下:- {
- let func = (anything: any) => {
- if (typeof anything === ‘string’) {
- return anything; // 类型是 string
- } elseif (typeof anything === ‘number’) {
- return anything; // 类型是 number
- }
- returnnull;
- };
- }
- {
- let func = (anything: string | number) => {
- if (typeof anything === ‘string’) {
- return anything; // 类型是 string
- } else {
- return anything; // 类型是 number
- }
- };
- }
- {
- type Goods = ‘pen’ | ‘pencil’ |‘ruler’;
- const getPenCost = (item: ‘pen’) =>2;
- const getPencilCost = (item: ‘pencil’) =>4;
- const getRulerCost = (item: ‘ruler’) =>6;
- const getCost = (item: Goods) => {
- if (item === ‘pen’) {
- return getPenCost(item); // item => ‘pen’
- } elseif (item === ‘pencil’) {
- return getPencilCost(item); // item => ‘pencil’
- } else {
- return getRulerCost(item); // item => ‘ruler’
- }
- }
- }
- const getCost = (item: Goods) => {
- if (item === ‘pen’) {
- item; // item => ‘pen’
- } else {
- item; // => ‘pencil’ | ‘ruler’
- }
- }