Skip to content
Permalink

Comparing changes

Choose two branches to see what’s changed or to start a new pull request. If you need to, you can also or learn more about diff comparisons.

Open a pull request

Create a new pull request by comparing changes across two branches. If you need to, you can also . Learn more about diff comparisons here.
base repository: luwenhua/typescript-tutorial
Failed to load repositories. Confirm that selected base ref is valid, then try again.
Loading
base: main
Choose a base ref
...
head repository: wangdoc/typescript-tutorial
Failed to load repositories. Confirm that selected head ref is valid, then try again.
Loading
compare: main
Choose a head ref
Able to merge. These branches can be automatically merged.
Loading
Showing with 324 additions and 171 deletions.
  1. +2 −2 .github/workflows/wangdoc.yml
  2. +1 −1 chapters.yml
  3. +1 −1 docs/any.md
  4. +1 −19 docs/assert.md
  5. +163 −45 docs/class.md
  6. +1 −1 docs/comment.md
  7. +21 −6 docs/declare.md
  8. +24 −16 docs/decorator.md
  9. +4 −5 docs/enum.md
  10. +10 −8 docs/function.md
  11. +2 −2 docs/interface.md
  12. +2 −2 docs/mapping.md
  13. +21 −19 docs/module.md
  14. +2 −3 docs/object.md
  15. +12 −13 docs/operator.md
  16. +50 −21 docs/tsconfig.json.md
  17. +5 −5 docs/utility.md
  18. +2 −2 package.json
4 changes: 2 additions & 2 deletions .github/workflows/wangdoc.yml
Original file line number Diff line number Diff line change
@@ -10,11 +10,11 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v3
uses: actions/checkout@v4
with:
persist-credentials: false
- name: Setup Node.js
uses: actions/setup-node@v3
uses: actions/setup-node@v4
with:
node-version: 'latest'
- name: Install dependencies
2 changes: 1 addition & 1 deletion chapters.yml
Original file line number Diff line number Diff line change
@@ -18,7 +18,7 @@
- decorator-legacy.md: 装饰器(旧语法)
- declare.md: declare 关键字
- d.ts.md: d.ts 类型声明文件
- operator.md: 运算符
- operator.md: 类型运算符
- mapping.md: 类型映射
- utility.md: 类型工具
- comment.md: 注释指令
2 changes: 1 addition & 1 deletion docs/any.md
Original file line number Diff line number Diff line change
@@ -240,7 +240,7 @@ let v2:string = f(); // 不报错
let v3:boolean = f(); // 不报错
```

上面示例中,函数`f()`会抛错,所以返回值类型可以写成`never`,即不可能返回任何值。各种其他类型的变量都可以赋值为`f()`的运行结果(`never`类型)。
上面示例中,函数`f()`会抛出错误,所以返回值类型可以写成`never`,即不可能返回任何值。各种其他类型的变量都可以赋值为`f()`的运行结果(`never`类型)。

为什么`never`类型可以赋值给任意其他类型呢?这也跟集合论有关,空集是任何集合的子集。TypeScript 就相应规定,任何类型都包含了`never`类型。因此,`never`类型是任何其他类型所共有的,TypeScript 把这种情况称为“底层类型”(bottom type)。

20 changes: 1 addition & 19 deletions docs/assert.md
Original file line number Diff line number Diff line change
@@ -114,15 +114,6 @@ const s2:string = value as string; // 正确

上面示例中,unknown 类型的变量`value`不能直接赋值给其他类型的变量,但是可以将它断言为其他类型,这样就可以赋值给别的变量了。

另外,类型断言也适合指定联合类型的值的具体类型。

```typescript
const s1:number|string = 'hello';
const s2:number = s1 as number;
```

上面示例中,变量`s1`是联合类型,可以断言其为联合类型里面的一种具体类型,再将其赋值给变量`s2`

## 类型断言的条件

类型断言并不意味着,可以把某个值断言为任意类型。
@@ -499,18 +490,9 @@ function assertIsDefined<T>(

上面示例中,工具类型`NonNullable<T>`对应类型`T`去除空类型后的剩余类型。

如果要将断言函数用于函数表达式,可以采用下面的写法。
如果要将断言函数用于函数表达式,可以采用下面的写法。根据 TypeScript 的[要求](https://github.com/microsoft/TypeScript/pull/33622#issuecomment-575301357),这时函数表达式所赋予的变量,必须有明确的类型声明。

```typescript
// 写法一
const assertIsNumber = (
value:unknown
):asserts value is number => {
if (typeof value !== 'number')
throw Error('Not a number');
};

// 写法二
type AssertIsNumber =
(value:unknown) => asserts value is number;

208 changes: 163 additions & 45 deletions docs/class.md
Original file line number Diff line number Diff line change
@@ -471,7 +471,7 @@ interface Swimmable {
// ...
}

interface SuperCar extends MotoVehicle,Flyable, Swimmable {
interface SuperCar extends MotorVehicle,Flyable, Swimmable {
// ...
}

@@ -570,7 +570,7 @@ const c1:Car = new Car();
const c2:MotorVehicle = new Car();
```

上面示例中,变量的类型可以写成类`Car`,也可以写成接口`MotorVehicle`。它们的区别是,如果类`Car`有接口`MotoVehicle`没有的属性和方法,那么只有变量`c1`可以调用这些属性和方法。
上面示例中,变量的类型可以写成类`Car`,也可以写成接口`MotorVehicle`。它们的区别是,如果类`Car`有接口`MotorVehicle`没有的属性和方法,那么只有变量`c1`可以调用这些属性和方法。

作为类型使用时,类名只能表示实例的类型,不能表示类的自身类型。

@@ -954,64 +954,47 @@ class Test extends getGreeterBase() {

上面示例中,例一和例二的`extends`关键字后面都是构造函数,例三的`extends`关键字后面是一个表达式,执行后得到的也是一个构造函数。

对于那些只设置了类型、没有初值的顶层属性,有一个细节需要注意。
## override 关键字

```typescript
interface Animal {
animalStuff: any;
}

interface Dog extends Animal {
dogStuff: any;
}

class AnimalHouse {
resident: Animal;
子类继承父类时,可以覆盖父类的同名方法。

constructor(animal:Animal) {
this.resident = animal;
```typescript
class A {
show() {
// ...
}
hide() {
// ...
}
}

class DogHouse extends AnimalHouse {
resident: Dog;

constructor(dog:Dog) {
super(dog);
class B extends A {
show() {
// ...
}
hide() {
// ...
}
}
```

上面示例中,类`DogHouse`的顶层成员`resident`只设置了类型(`Dog`),没有设置初值。这段代码在不同的编译设置下,编译结果不一样。

如果编译设置的`target`设成大于等于`ES2022`,或者`useDefineForClassFields`设成`true`,那么下面代码的执行结果是不一样的。

```typescript
const dog = {
animalStuff: 'animal',
dogStuff: 'dog'
};

const dogHouse = new DogHouse(dog);

console.log(dogHouse.resident) // undefined
```

上面示例中,`DogHouse`实例的属性`resident`输出的是`undefined`,而不是预料的`dog`。原因在于 ES2022 标准的 Class Fields 部分,与早期的 TypeScript 实现不一致,导致子类的那些只设置类型、没有设置初值的顶层成员在基类中被赋值后,会在子类被重置为`undefined`,详细的解释参见《tsconfig.json》一章,以及官方 3.7 版本的[发布说明](https://www.typescriptlang.org/docs/handbook/release-notes/typescript-3-7.html#the-usedefineforclassfields-flag-and-the-declare-property-modifier)
上面示例中,B 类定义了自己的`show()`方法和`hide()`方法,覆盖了 A 类的同名方法。

解决方法就是使用`declare`命令,去声明顶层成员的类型,告诉 TypeScript 这些成员的赋值由基类实现
但是有些时候,我们继承他人的类,可能会在不知不觉中,就覆盖了他人的方法。为了防止这种情况,TypeScript 4.3 引入了 [override 关键字](https://www.typescriptlang.org/docs/handbook/release-notes/typescript-4-3.html#override-and-the---noimplicitoverride-flag)

```typescript
class DogHouse extends AnimalHouse {
declare resident: Dog;

constructor(dog:Dog) {
super(dog);
class B extends A {
override show() {
// ...
}
override hide() {
// ...
}
}
```

上面示例中,`resident`属性的类型声明前面用了`declare`命令,这样就能确保在编译目标大于等于`ES2022`时(或者打开`useDefineForClassFields`时),代码行为正确。
上面示例中,B 类的`show()`方法和`hide()`方法前面加了 override 关键字,明确表明作者的意图,就是要覆盖 A 类里面的这两个同名方法。这时,如果 A 类没有定义自己的`show()`方法和`hide()`方法,就会报错。

但是,这依然没有解决,子类无意中覆盖父类同名方法的问题。因此,TypeScript 又提供了一个编译参数`noImplicitOverride`。一旦打开这个参数,子类覆盖父类的同名方法就会报错,除非使用了 override 关键字。

## 可访问性修饰符

@@ -1278,6 +1261,140 @@ class A {
}
```

## 顶层属性的处理方法

对于类的顶层属性,TypeScript 早期的处理方法,与后来的 ES2022 标准不一致。这会导致某些代码的运行结果不一样。

类的顶层属性在 TypeScript 里面,有两种写法。

```typescript
class User {
// 写法一
age = 25;

// 写法二
constructor(private currentYear: number) {}
}
```

上面示例中,写法一是直接声明一个实例属性`age`,并初始化;写法二是顶层属性的简写形式,直接将构造方法的参数`currentYear`声明为实例属性。

TypeScript 早期的处理方法是,先在顶层声明属性,但不进行初始化,等到运行构造方法时,再完成所有初始化。

```typescript
class User {
age = 25;
}

// TypeScript 的早期处理方法
class User {
age: number;

constructor() {
this.age = 25;
}
}
```

上面示例中,TypeScript 早期会先声明顶层属性`age`,然后等到运行构造函数时,再将其初始化为`25`

ES2022 标准里面的处理方法是,先进行顶层属性的初始化,再运行构造方法。这在某些情况下,会使得同一段代码在 TypeScript 和 JavaScript 下运行结果不一致。

这种不一致一般发生在两种情况。第一种情况是,顶层属性的初始化依赖于其他实例属性。

```typescript
class User {
age = this.currentYear - 1998;

constructor(private currentYear: number) {
// 输出结果将不一致
console.log('Current age:', this.age);
}
}

const user = new User(2023);
```

上面示例中,顶层属性`age`的初始化值依赖于实例属性`this.currentYear`。按照 TypeScript 的处理方法,初始化是在构造方法里面完成的,会输出结果为`25`。但是,按照 ES2022 标准的处理方法,初始化在声明顶层属性时就会完成,这时`this.currentYear`还等于`undefined`,所以`age`的初始化结果为`NaN`,因此最后输出的也是`NaN`

第二种情况与类的继承有关,子类声明的顶层属性在父类完成初始化。

```typescript
interface Animal {
animalStuff: any;
}

interface Dog extends Animal {
dogStuff: any;
}

class AnimalHouse {
resident: Animal;

constructor(animal:Animal) {
this.resident = animal;
}
}

class DogHouse extends AnimalHouse {
resident: Dog;

constructor(dog:Dog) {
super(dog);
}
}
```

上面示例中,类`DogHouse`继承自`AnimalHouse`。它声明了顶层属性`resident`,但是该属性的初始化是在父类`AnimalHouse`完成的。不同的设置运行下面的代码,结果将不一致。

```typescript
const dog = {
animalStuff: 'animal',
dogStuff: 'dog'
};

const dogHouse = new DogHouse(dog);

console.log(dogHouse.resident) // 输出结果将不一致
```

上面示例中,TypeScript 的处理方法,会使得`resident`属性能够初始化,所以输出参数对象的值。但是,ES2022 标准的处理方法是,顶层属性的初始化先于构造方法的运行。这使得`resident`属性不会得到赋值,因此输出为`undefined`

为了解决这个问题,同时保证以前代码的行为一致,TypeScript 从3.7版开始,引入了编译设置`useDefineForClassFields`。这个设置设为`true`,则采用 ES2022 标准的处理方法,否则采用 TypeScript 早期的处理方法。

它的默认值与`target`属性有关,如果输出目标设为`ES2022`或者更高,那么`useDefineForClassFields`的默认值为`true`,否则为`false`。关于这个设置的详细说明,参见官方 3.7 版本的[发布说明](https://www.typescriptlang.org/docs/handbook/release-notes/typescript-3-7.html#the-usedefineforclassfields-flag-and-the-declare-property-modifier)

如果希望避免这种不一致,让代码在不同设置下的行为都一样,那么可以将所有顶层属性的初始化,都放到构造方法里面。

```typescript
class User {
age: number;

constructor(private currentYear: number) {
this.age = this.currentYear - 1998;
console.log('Current age:', this.age);
}
}

const user = new User(2023);
```

上面示例中,顶层属性`age`的初始化就放在构造方法里面,那么任何情况下,代码行为都是一致的。

对于类的继承,还有另一种解决方法,就是使用`declare`命令,去声明子类顶层属性的类型,告诉 TypeScript 这些属性的初始化由父类实现。

```typescript
class DogHouse extends AnimalHouse {
declare resident: Dog;

constructor(dog:Dog) {
super(dog);
}
}
```

上面示例中,`resident`属性的类型声明前面用了`declare`命令。这种情况下,这一行代码在编译成 JavaScript 后就不存在,那么也就不会有行为不一致,无论是否设置`useDefineForClassFields`,输出结果都是一样的。

## 静态成员

类的内部可以使用`static`关键字,定义静态成员。
@@ -1595,4 +1712,5 @@ class FileSystemObject {
## 参考链接

- [TypeScript Constructor in Interface](http://fritzthecat-blog.blogspot.com/2018/06/typescript-constructor-in-interface.html)
- [TypeScript: useDefineForClassFields – How to avoid future Breaking Changes](https://angular.schule/blog/2022-11-use-define-for-class-fields)

2 changes: 1 addition & 1 deletion docs/comment.md
Original file line number Diff line number Diff line change
@@ -55,8 +55,8 @@ function doStuff(abc: string, xyz: string) {
// do some stuff
}

// @ts-expect-error
expect(() => {
// @ts-expect-error
doStuff(123, 456);
}).toThrow();
```
Loading