外观
TypeScript-关键字与运算符
3881字约13分钟
2024-07-31
TypeScript中的运算符
typeof 运算符
在 TypeScript 中, typeof
运算符用于获取一个值的类型,并返回一个字符串表示该类型的名称。这与 JavaScript 中的行为类似,但在 TypeScript 中这个功能还可以被用来推断变量或表达式的类型。
- JavaScript 中的
typeof
运算符
console.log(typeof 5); // "number"
console.log(typeof "hello"); // "string"
console.log(typeof true); // "boolean"
console.log(typeof undefined); // "undefined"
console.log(typeof null); // "object" (这是一个已知的特殊情况)
console.log(typeof []); // "object"
console.log(typeof {}); // "object"
console.log(typeof 123n) // "bigint"
console.log(typeof Symbol("name")) // "symbol"
console.log(typeof parseInt); // "function"
- TypeScript 中的
typeof
运算符
let myValue = 10;
type MyValueType = typeof myValue; // number
const a = { x: 0 };
type T0 = typeof a; // { x: number }
type T1 = typeof a.x; // number
let a = 1;
let b: typeof a;
if (typeof a === "number") {
b = a;
}
// 上面转成JS后
let a = 1;
let b;
if (typeof a === "number") {
b = a;
}
从上面的示例可以看出:
typeof
运算符可以在运行时获取变量的类型。转成 Javascript 后,typeof
运算符会保留。- 在 TypeScript 中,
typeof
还可以用于获取变量的类型并在类型注解中使用,增强了类型的复用性和代码的可读性。用于 TypeScript 类型相关的代码,会在转成 Javascript 后,typeof
运算符会被删除。
keyof 运算符
在 TypeScript 中, keyof
运算符用于获取一个对象类型的所有键,并返回一个联合类型。它可以用于类型推断和类型检查。
基本用法
- 获取对象类型的所有键,并返回一个联合类型。
interface Person {
name: string;
age: number;
}
type PersonKeys = keyof Person; // "name" | "age"
- 对于没有键的对象类型,
keyof
返回的类型是never
。
type EmptyObject = {};
type EmptyObjectKeys = keyof EmptyObject; // never
- 如果对象的属性名采用索引签名,则
keyof
返回的类型是索引签名的类型。
interface StringArray {
[index: number]: string;
}
type StringArrayKeys = keyof StringArray; // number
// 属性名为string时,包含了属性名为number的情况,因为数值会自动转为字符串。
// 因此下面输出的是string | number
interface T {
[prop: string]: number;
}
type TKeys = keyof T; // string | number
- 对于联合类型,
keyof
返回共有键名的联合类型。
interface Teacher {
name: string;
subject: string;
}
type Person = {
name: string;
age: number;
}
type T = Person | Teacher;
type TKeys = keyof T; // "name"
- 对于交叉类型,
keyof
返回的键名的联合类型。
// 如上定义
type T = Person & Teacher;
type TKeys = keyof T; // "name" | "age" | "subject"
- keyof 与映射类型,对象类型中的每个属性类型进行映射。
interface Person {
name: string;
age: number;
}
type ReadonlyPerson = {
readonly [K in keyof Person]: Person[K];
};
const person: ReadonlyPerson = {
name: "Alice",
age: 30
};
注意事项
keyof
返回的联合类型中的每个成员都是对象类型的键,并且这些键的类型是string | number | symbol
之一。
使用技巧
- 利用
keyof
取出对象所有值类型
interface Person {
name: string;
age: number;
}
type PersonValues = Person[keyof Person]; // string | number
- 利用
keyof
去掉或者添加对象的readonly
修饰符。
interface Person {
readonly name: string;
readonly age: number;
}
type MutablePerson = {
-readonly [K in keyof Person]: Person[K];
};
type ReadonlyPerson = {
+readonly [K in keyof Person]: Person[K];
};
如上,-readonly
表示去掉 readonly
修饰符,同理+readonly
则为添加 readonly
修饰符。
in 运算符
在 JavaScript 中, in
运算符用于检查一个属性是否存在于对象中。
const obj = { a: 1, b: 2, c: 3 };
console.log("a" in obj); // true
console.log("d" in obj); // false
而在 TypeScript 中, in
运算符用来遍历联合类型的每个成员。
type Colors = "red" | "green" | "blue";
type ColorMap = {
[K in Colors]: string; // 依次遍历联合类型中的每个成员
};
// 等价于
type ColorMap = {
red: string;
green: string;
blue: string;
};
is 运算符
在 TypeScript 中,is 运算符通常用于类型保护,帮助 TypeScript 确定某个变量的具体类型。它通常与用户自定义的类型保护函数一起使用。
is
运算符用来描述返回值属于 true
还是 false
。
基础示例:
function isString(value: any): value is string {
return typeof value === 'string';
}
function processValue(value: any) {
if (isString(value)) {
console.log(`Value is a string: ${value.toUpperCase()}`);
} else {
console.log(`Value is not a string`);
}
}
[] 访问运算符
在 TypeScript 类型操作中,[]
访问运算符用于获取对象类型中指定键的属性类型。
- 获取对象类型中指定键的属性类型
interface Person {
name: string;
age: number;
}
type NameType = Person["name"]; // string
type AgeType = Person["age"]; // number
[]
中的参数可以是联合类型,表示获取对象类型中多个键的属性类型,返回的也是联合类型。
type NameOrAge = Person["name" | "age"]; // string | number
[]
中的参数可以是属性名的索引类型。
type Person = {
[key: string]: number;
}
type PersonValues = Person[string]; // number
注意事项
[]
中的参数必须是对象类型的键,否则会报错。
extends...? 条件运算符
在 TypeScript 中, extends...?
是一个条件运算符,用于根据条件判断返回的类型。
type IsString<T> = T extends string ? true : false;
如上,IsString
是一个泛型类型,接受一个类型参数 T
,如果 T
是 string
类型,则返回 true
,否则返回 false
。
联合类型也可以使用条件运算符。
type IsStringOrNumber<T> = T extends string | number ? true : false;
模版字符串
在 TypeScript 中,可以使用模版字符串来定义类型。
type Name = "Alice";
type Greeting = `Hello, ${Name}!`; // "Hello, Alice!"
如果模版字符串中是一个联合类型,那么返回的类型也是一个联合类型。
type Name = "Alice" | "Bob";
type Greeting = `Hello, ${Name}!`; // "Hello, Alice!" | "Hello, Bob!"
注意事项
- 模版字符串中的变量必须是类型,不能是值。
- 模版字符串可以引用的类型:
string
、number
、bigint
、boolean
、null
、undefined
。引用其他类型会报错。
TypeScript 中的关键字
type 关键字
在 TypeScript 中, type
关键字用于定义类型别名。类型别名可以是基本类型、联合类型、交叉类型、元组、对象类型等等使用 type
可以让代码更具可读性和可维护性。
基本语法
type 类型别名 = 类型;
注意事项
- 不可拓展
- 不能重新定义,同一作用域内不能重名
- 允许嵌套,在一个
type
里使用另一个type
示例:
- 基本类型别名
type StringOrNumber = string | number;
let value: StringOrNumber;
value = "Hello"; // 合法
value = 42; // 合法
valeue = true; // 非法 报错
- 函数类型别名
type GreetFunction = (name: string) => string;
const greet: GreetFunction = (name) => {
return `Hello, ${name}!`;
};
console.log(greet("Bob")); // 输出: Hello, Bob!
- 对象类型别名
type Person = {
name: string;
age: number;
};
const person: Person = {
name: "Alice",
age: 30
};
- 联合类型别名
type Status = "success" | "error" | "loading";
let currentStatus: Status;
currentStatus = "success"; // 合法
currentStatus = "failed"; // 错误: 不能将类型“"failed"”分配给类型“Status”。
- 交叉类型别名
type Person = {
name: string;
age: number;
};
type Address = {
street: string;
city: string;
};
type PersonWithAddress = Person & Address;
const personWithAddress: PersonWithAddress = {
name: "Alice",
age: 30,
street: "123 Main St",
city: "Wonderland"
};
interface 关键字
在 TypeScript 中, interface
关键字用于定义对象类型。它可以用来描述对象的结构和行为,也可以用来定义函数类型。
提示
关于 interface
的详细内容可以查看:TypeScript-interface
as 关键字
在 TypeScript 中,as
关键字用于类型断言(Type Assertion),它允许开发者告诉编译器某个值的具体类型。它可以将一个值的类型断言为另一个类型。
- 基本用法
let myValue: any = 10;
let myNumber: number = myValue as number; // 类型断言
- 断言接口
interface Person {
name: string;
age: number;
}
let person: any = { name: "John", age: 30 };
let john = person as Person;
需要注意的是,除了 as
关键字外,TypeScript 还支持另一种类型断言的语法,即使用尖括号。
- 不过在 JSX 中,尖括号语法可能会与 HTML 标签冲突,因此推荐使用
as
语法。
let myValue: any = 10;
let myNumber: number = <number>myValue; // 类型断言
interface Person {
name: string;
age: number;
}
let person: any = { name: "John", age: 30 };
let john = <Person>person;
implements 关键字
在 TypeScript 中, implements
关键字用于实现接口(interface)。当一个类实现一个接口时,它必须提供接口中定义的所有属性和方法的具体实现。这种机制使得 TypeScript 支持面向对象编程中的多态性和接口的概念。
- 基本用法
// 定义一个接口
interface Animal {
name: string;
sound(): void;
}
// 实现接口的类
class Dog implements Animal {
name: string;
constructor(name: string) {
this.name = name;
}
sound() {
console.log(`${this.name} says Woof!`);
}
}
// 创建实例
const dog = new Dog('Buddy');
dog.sound(); // 输出: Buddy says Woof!
- 多个接口实现。一个类可以实现多个接口,类必须提供所有接口中定义的属性和方法。
interface CanRun {
run(): void;
}
interface CanBark {
bark(): void;
}
class Dog implements CanRun, CanBark {
name: string;
constructor(name: string) {
this.name = name;
}
run() {
console.log(`${this.name} is running.`);
}
bark() {
console.log(`${this.name} says Woof!`);
}
}
const dog = new Dog('Buddy');
dog.run(); // 输出: Buddy is running.
dog.bark(); // 输出: Buddy says Woof!
- 接口的继承
接口可以继承其他接口,类在实现时需要实现所有继承的接口中的属性和方法。
interface Animal {
name: string;
}
interface Dog extends Animal {
bark(): void;
}
class Beagle implements Dog {
name: string;
constructor(name: string) {
this.name = name;
}
bark() {
console.log(`${this.name} says Woof!`);
}
}
const beagle = new Beagle('Max');
beagle.bark(); // 输出: Max says Woof!
namespace 关键字
在 TypeScript 中, namespace
关键字用于定义命名空间。命名空间是一种组织和隔离代码的方式,它可以将相关的代码组织到一个范围内。
declare 关键字
在 TypeScript 中, declare
关键字用于声明全局变量、全局函数、全局类型等。它可以帮助编译器识别这些声明,从而提供更好的类型检查和代码补全功能。换句话说 declare
关键字用来告诉编译器,某个类型是存在的,可以在当前文件中使用。
它的主要作用,就是让当前文件可以使用其他文件声明的类型。举例来说,自己的脚本使用外部库定义的函数,编译器会因为不知道外部函数的类型定义而报错,这时就可以在自己的脚本里面使用
declare
关键字,告诉编译器外部函数的类型。这样的话,编译单个脚本就不会因为使用了外部类型而报错。
基础示例
// 直接使用其他文件定义的类型会报错
globalVariable = 'Hello' // 报错
// 可以使用declare 命令给出它的类型,就不会报错。
declare const globalVariable: string;
globalVariable = 'Hello'; // 正确
如上所示,使用 declare
命令可以告诉编译器,某个类型是存在的,可以在当前文件中使用。
需要注意的是:
declare
关键字,它只是通知编译器某个类型是存在的,不用给出具体实现。比如,只描述函数的类型,不给出函数的实现,如果不使用declare
,这是做不到的。declare
只能用来描述已经存在的变量和数据结构,不能用来声明新的变量和数据结构。另外,所有declare
语句都不会出现在编译后的文件里面。
描述类型
- 变量
- type 或者 interface 声明的类型
- 类(class)
- 枚举(enum)
- 函数(function)
- 模块(module)
- 命名空间(namespace)
变量
declare
关键字可以给出外部变量的类型描述。例如:当前的脚本中使用了其他文件定义的全局变量,但是编译器不知道这个变量的类型,这时可以使用 declare
关键字来描述这个变量的类型。
declare var globalVariable: string;
globalVariable = 'Hello';
注意:declare
不能设置变量的值,只能描述变量的类型。
declare var globalVariable: string = 'Hello'; // 报错
函数
declare
关键字可以给出外部函数的类型描述。然后我们就可以直接使用这个函数,而不用关心它的具体实现。
declare function greet(name: string): void;
greet('Alice');
类(Class)
declare
关键字可以给出外部类的类型描述。
declare class Person {
name: string;
age: number;
constructor(name: string, age: number);
sayHello(): void;
}
模块和命名空间
declare
关键字可以给出外部模块和命名空间的类型描述。下面示例中的export
关键字可以省略。
declare module 'moduleName' {
export interface MyInterface {
property: string;
method(): void;
}
}
declare namespace MyNamespace {
export const myConstant: number;
export function myFunction(): void;
}
举例说明,下面例子中使用了 myDemoLib
外部库定义的类型,但是编译器不知道这个类型的具体实现,这时可以使用 declare
关键字来描述这个类型。
// 类型描述
declare namespace myDemoLib {
export function sayHello(message: string): string;
export const count: number;
}
// 外部库使用
let result = myDemoLib.sayHello("你好");
let count = myDemoLib.count;
Global
如果要为 JavaScript 引擎的原生对象添加属性和方法,可以使用 declare global {}
语法。
declare global {
interface String {
reverse(): string;
}
}
String.prototype.reverse = function() {
return this.split('').reverse().join('');
};
console.log("hello".reverse()); // olleh
给 window
对象添加属性和方法。
declare global {
interface Window {
myProperty: string;
}
}
window.myProperty = "Hello";
console.log(window.myProperty); // Hello
枚举(Enum)
declare
关键字可以给出外部枚举的类型描述。
declare enum Color {
Red,
Green,
Blue
}
类型声明文件
类型声明文件通常以 .d.ts
为后缀,用于描述 JavaScript 库的类型信息。类型声明文件中通常包含 declare
关键字来描述变量、函数、类、模块等。
declare module "url" {
export interface Url {
protocol?: string;
hostname?: string;
pathname?: string;
}
export function parse(
urlStr: string,
parseQueryString?,
slashesDenoteHost?
): Url;
}
declare module "path" {
export function normalize(p: string): string;
export function join(...paths: any[]): string;
export var sep: string;
}
使用类型声明文件时,需要使用 /// <reference path="name.d.ts" />
语法来引用类型声明文件。
abstract 关键字
在 TypeScript 中, abstract
关键字用于定义抽象类和抽象方法。抽象类是不能被实例化的类,只能被继承。抽象成员是抽象类中定义的成员,必须在子类中实现。
abstract class Person {
abstract name: string;
abstract sayHello(): void;
}
class Student extends Person {
name: string = "yuwb";
sayHello(): void {
console.log(`Hello, my name is ${this.name}`);
}
}
const student = new Student();
student.sayHello(); // Hello, my name is yuwb
asserts 关键字
在 TypeScript 中,asserts 关键字用于类型保护,允许你在函数中进行类型断言,以确保某个条件为真时,特定的类型是有效的。这通常用于自定义类型保护函数,以便在运行时检查某个值的类型,并在检查通过后,告知 TypeScript 编译器该值的类型。
类型保护函数示例:
// 定义一个类型
type User = {
name: string;
age: number;
};
// 自定义类型保护函数
function assertIsUser(user: any): asserts user is User {
if (typeof user !== 'object' || user === null) {
throw new Error('Not a user');
}
if (typeof user.name !== 'string' || typeof user.age !== 'number') {
throw new Error('Invalid user properties');
}
}
// 使用类型保护函数
function processUser(user: any) {
assertIsUser(user); // 运行时检查
// 这里 TypeScript 知道 user 是 User 类型
console.log(`User name: ${user.name}, age: ${user.age}`);
}
// 测试
const userInput: any = { name: 'Alice', age: 30 };
processUser(userInput); // 正常工作
const invalidInput: any = { name: 'Bob', age: 'thirty' };
processUser(invalidInput); // 抛出错误
换句话说,该函数用来断言参数 user
的类型是 User
,如果参数 user
的类型不是 User
,则抛出错误。程序就会在这里中断。如果达到要求,则 user
的类型会被断言为 User
,从而在后续代码中可以安全地使用 User
类型的属性和方法。
infer 关键字
在 TypeScript 中,infer
关键字用于在条件类型中推断类型。它通常与泛型一起使用,以便在类型转换或类型推导过程中获取中间类型。
基础示例:
type Flatten<Type> = Type extends Array<infer Item> ? Item : Type;
// string
type Str = Flatten<string[]>;
// number
type Num = Flatten<number>;
如上,Flatten
类型定义了一个条件类型,如果 Type
是一个数组类型,则推断出数组元素的类型 Item
,否则返回 Type
本身。infer
关键字在这里用于推断数组元素的类型。