【Typescript入门手册】类型进阶

简介: 【Typescript入门手册】类型进阶

【Typescript专题】之类型进阶
🚀【TypeScript入门手册】记录了出场率较高的Ts概念,旨在帮助大家了解并熟悉Ts
🎉 本系列会持续更新并更正,重点关照大家感兴趣的点,欢迎同学留言交流,在进阶之路上,共勉!
👍 star本项目给作者一点鼓励吧
📚 系列文章,收藏 不走丢哦

一、类型别名(type)
类型别名用来给一个类型起个新名字。例如:

type isNumber = number;
const num: isNumber = 1;
1
2
上面的例子没有任何问题,当然也是一句“废话”,那么类型别名又是为什么创造的呢?来看下面的例子:

type Name = string; // 字符串
type NameResolver = () => string; // 函数
type NameOrResolver = Name | NameResolver; // 联合类型

function getName(n: NameOrResolver): Name {
if (typeof n === "string") {

return n;

} else {

return n();

}
}
1
2
3
4
5
6
7
8
9
10
11
别名常用于联合类型,如:

type ID = number | string;
const id1: ID = 123;
const id2: ID = "wpsd";
1
2
3

二、属性修饰符(Property Modifiers)
对象类型中的每个属性可以说明它的类型、属性是否可选、属性是否只读等信息。

2.1 可选属性(Optional Properties)
我们可以在属性名后面加一个?标记表示这个属性是可选的:

type Shape = "circle" | "square";
interface PaintOptions {

shape: Shape;
xPos?: number;
yPos?: number;

}

function paintShape(opts: PaintOptions) {

let xPos = opts.xPos; // (property) PaintOptions.xPos?: number
let yPos = opts.yPos; // (property) PaintOptions.yPos?: number

}

const shape = 'circle';

paintShape({ shape });
paintShape({ shape, xPos: 100 });
paintShape({ shape, yPos: 100 });
paintShape({ shape, xPos: 100, yPos: 100 });
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
在这个例子中,xPos 和 yPos 就是可选属性。因为他们是可选的,所以上面所有的调用方式都是合法的。

在 JavaScript 中,如果一个属性值没有被设置,我们获取会得到 undefined 。所以我们可以针对 undefined 特殊处理一下——解构

function paintShape({ shape, xPos = 0, yPos = 0 }: PaintOptions) {

console.log("x coordinate at", xPos); // (parameter) xPos: number
console.log("y coordinate at", yPos); // (parameter) yPos: number
// ...

}
1
2
3
4
5
这里我们使用了解构语法以及为 xPos 和 yPos 提供了默认值。现在 xPos 和 yPos 的值在 paintShape 函数内部一定存在,但对于 paintShape 的调用者来说,却是可选的。

注意:现在并没有在解构语法里放置类型注解的方式。这是因为在 JavaScript 中,下面的语法代表的意思完全不同。

function draw({ shape: Shape, xPos: number = 100 /.../ }) {

render(shape);
// 找不到名称“shape”。你是否指的是“Shape”?
render(xPos);
// 找不到名称“xPos”

}
1
2
3
4
5
6
在对象解构语法中,shape: Shape 表示的是把shape的值赋值给局部变量 Shape。 xPos: number也是一样,会基于xPos创建一个名为number的变量。

2.2 readonly 属性(readonly Properties)
在TypeScript中,属性可以被标记为·readonly·,这不会改变任何运行时的行为,但在类型检查的时候,一个标记为readonly的属性是不能被写入的。

interface SomeType {

readonly prop: string;

}
function doSomething(obj: SomeType) {

console.log(`prop has the value '${obj.prop}'.`);

obj.prop = "hello";
// 无法分配到 "prop" ,因为它是只读属性

}
1
2
3
4
5
6
7
8
9
不过使用readonly并不意味着一个值就完全是不变的,亦或者说,内部的内容是不能变的。readonly仅仅表明属性本身是不能被重新写入的。大家应该猜到了,如果是引用类型,则可以避开这个问题。

interface Developer {

readonly fe: { name: string; age: number };

}

function getDeveloper(developer: Developer) {

console.log(developer.fe.name);
// (property) name: string
developer.fe.name = '余光';
developer.fe.age++;

}

function getDeveloper1(developer: Developer) {

console.log(developer.fe.name);
// (property) name: string
developer.fe = {
    name: "余光",
    age: 18,
};
// 无法分配到 "fe" ,因为它是只读属性。ts(2540)

}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
TypeScript 在检查两个类型是否兼容的时候,并不会考虑两个类型里的属性是否是readonly,这就意味着,readonly 的值是可以通过别名修改的。

interface Person {

name: string;
age: number;

}

interface ReadonlyPerson {

readonly name: string;
readonly age: number;

}

let writablePerson: Person = {

name: "Person McPersonface",
age: 42,

};

// works
let readonlyPerson: ReadonlyPerson = writablePerson;

console.log(readonlyPerson.age); // prints '42'
writablePerson.age++;
console.log(readonlyPerson.age); // prints '43'
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
2.3 索引签名(Index Signatures)
有的时候,你不能提前知道一个类型里的所有属性的名字,但是你知道这些值的特征。

这种情况,你就可以用一个索引签名 (index signature) 来描述可能的值的类型,举个例子:

interface StringArray {

[index: number]: string;

}

const myArray: StringArray = [1, 2, 3];// ❌ 不能将类型“number”分配给类型“string”
const secondItem = myArray[1]; // const secondItem: string
1
2
3
4
5
6
这样,我们就有了一个具有索引签名的接口StringArray,一个索引签名的属性类型必须是 string 或者是 number。

三、属性继承(Extending Types)
有时我们需要一个比其他类型更具体的类型。举个例子,假设我们有一个BasicGoods类型用来描述一个商品的基本信息

interface BasicGoods {

color: string;
size: string;
brand: string;
address: string;

}
1
2
3
4
5
6
这在一些情况下已经满足了,但同一个品牌的商品可能在,不同的分类中,例如:

interface BasicGoodsWithCategory {

color: string;
size: string;
brand: string;
address: string;
category: string

}
1
2
3
4
5
6
7
这样写固然可以,但为了加一个字段,就是要完全的拷贝一遍。我们可以改成继承BasicGoods的方式来实现:

interface BasicGoodsWithCategory extends BasicGoods{

category: string

}
1
2
3
对接口使用extends关键字允许我们有效的从其他声明过的类型中拷贝成员,并且随意添加新成员。

接口也可以继承多个类型:

interface Colorful {

color: string;

}

interface Circle {

radius: number;

}

interface ColorfulCircle extends Colorful, Circle {}

const cc: ColorfulCircle = {

color: "red",
radius: 42,

};
1
2
3
4
5
6
7
8
9
10
11
12
13
14

四、交叉类型(Intersection Types)
TypeScrip 也提供了名为交叉类型的方法,用于合并已经存在的对象类型。交叉类型的定义需要用到&操作符:

interface Colorful {

color: string;

}
interface Circle {

radius: number;

}

type group = Colorful & Circle;

function draw(circle: group) {

console.log(`Color was ${circle.color}`); // (property) Circle.radius: number
console.log(`Radius was ${circle.radius}`); // (property) Circle.radius: number

}
1
2
3
4
5
6
7
8
9
10
11
12
13
这里,我们连结Colorful和Circle产生了一个新的类型,新类型拥有Colorful和Circle的所有成员。

组合和继承,在上面的两个例子中解决的问题是一样的,那么怎么区分呢?

五、接口继承与交叉类型(Interfalces vs Intersections)
这两种方式在合并类型上看起来很相似,但实际上还是有很大的不同。最原则性的不同就是在于冲突怎么处理,这也是你决定选择那种方式的主要原因。

interface Colorful {

color: string;

}
interface ColorfulSub extends Colorful {

color: number;

}
// ❌
// 接口“ColorfulSub”错误扩展接口“Colorful”。
// 属性“color”的类型不兼容。
// 不能将类型“number”分配给类型“string”。
1
2
3
4
5
6
7
8
9
10
使用继承的方式,如果重写类型会导致编译错误,但交叉类型不会:

interface Colorful {

color: string;

}

type ColorfulSub = Colorful & {

color: number;

};
1
2
3
4
5
6
7

虽然不会报错,那 color 属性的类型是什么呢,答案是never,取得是string和number的交集。

六、泛型对象类型(Generic Object Types)
让我们写这样一个Box类型,可以包含任何值,此时需要做一些预防检查,或者用一个容易错误的类型断言。

interface Box {

contents: unknown;

}

let x: Box = {

contents: "hello world",

};

// we could check 'x.contents'
if (typeof x.contents === "string") {

console.log(x.contents.toLowerCase());

}

// or we could use a type assertion
console.log((x.contents as string).toLowerCase());
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
一个更加安全的做法是将 Box 根据 contents 的类型拆分的更具体一些:

interface NumberBox {

contents: number;

}

interface StringBox {

contents: string;

}

interface BooleanBox {

contents: boolean;

}
1
2
3
4
5
6
7
8
9
10
11
但是这也意味着我们不得不创建不同的函数或者函数重载处理不同的类型:

function setContents(box: StringBox, newContents: string): void;
function setContents(box: NumberBox, newContents: number): void;
function setContents(box: BooleanBox, newContents: boolean): void;
function setContents(box: { contents: any }, newContents: any) {

box.contents = newContents;

}
1
2
3
4
5
6
这样写就太繁琐了。此时引入一个概念——泛型,反省Box ,它声明了一个类型参数 (type parameter):

interface Box {

contents: Type;

}
1
2
3
你可以这样理解:Box的Type就是contents拥有的类型Type。当我们引用Box的时候,我们需要给予一个类型实参替换掉Type:

let aaa: Box = {

contents: 1

};
// ❌ 不能将类型“number”分配给类型“string”。
1
2
3
4
把 Box 想象成一个实际类型的模板,Type 就是一个占位符,可以被替代为具体的类型。当 TypeScript 看到 Box,它就会替换为 Box 的 Type 为 string ,最后的结果就会变成 { contents: string }。换句话说,Box和 StringBox 是一样的。

interface Box {
contents: Type;
}
interface StringBox {
contents: string;
}
1
2
3
4
5
6
不过现在的 Box 是可重复使用的,如果我们需要一个新的类型,我们完全不需要再重新声明一个类型。

interface Box {
contents: Type;
}

interface Apple {
// ....
}

// Same as '{ contents: Apple }'.
type AppleBox = Box;
// 这也意味着我们可以利用泛型函数避免使用函数重载。

function setContents(box: Box, newContents: Type) {
box.contents = newContents;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
6.1 类型别名与泛型
interface Box {

contents: Type;

}
1
2
3
使用别名对应就是:

type Box = {

contents: Type;

};
1
2
3
类型别名不同于接口,可以描述的不止是对象类型,所以我们也可以用类型别名写一些其他种类的的泛型帮助类型。

type OrNull = Type | null;

type OneOrMany = Type | Type[];

type OneOrManyOrNull = OrNull<OneOrMany>;

type OneOrManyOrNull = OneOrMany | null

type OneOrManyOrNullStrings = OneOrManyOrNull;

type OneOrManyOrNullStrings = OneOrMany | null
1
2
3
4
5
6
7
8
9
10
11
现代 JavaScript 也提供其他是泛型的数据结构,比如 Map<K, V> , Set 和 Promise。因为 Map 、Set 、Promise的行为表现,它们可以跟任何类型搭配使用。

七、字符串字面量类型
字符串字面量类型用来约束取值只能是某几个字符串中的一个。

type EventNames = "click" | "scroll" | "mousemove";
function handleEvent(ele: Element, event: EventNames) {
// do something
}

handleEvent(document.getElementById("hello"), "scroll"); // 没问题
handleEvent(document.getElementById("world"), "onmouseout"); // 报错,event 不能为 'onmouseout'
1
2
3
4
5
6
7
上例中,我们使用 type 定了一个字符串字面量类型 EventNames,它只能取三种字符串中的一种。

写在最后
本篇文章是《Typescript基础入门》第四篇,本篇文章主要聊一聊基本类型相关的扩展,毕竟随着文章的加深,我在阅读官方文档的时候经常会见到陌生的声明和“单词”,感觉有必要跟大家分享一下,下一篇文章,我们主要聊一聊Ts的泛型。

相关文章
|
2月前
|
设计模式 JavaScript 安全
TypeScript性能优化及代码质量提升的重要性、方法与策略,包括合理使用类型注解、减少类型断言、优化模块导入导出、遵循编码规范、加强代码注释等
本文深入探讨了TypeScript性能优化及代码质量提升的重要性、方法与策略,包括合理使用类型注解、减少类型断言、优化模块导入导出、遵循编码规范、加强代码注释等,旨在帮助开发者在保证代码质量的同时,实现高效的性能优化,提升用户体验和项目稳定性。
53 6
|
2月前
|
开发框架 JavaScript 前端开发
TypeScript 是一种静态类型的编程语言,它扩展了 JavaScript,为 Web 开发带来了强大的类型系统、组件化开发支持、与主流框架的无缝集成、大型项目管理能力和提升开发体验等多方面优势
TypeScript 是一种静态类型的编程语言,它扩展了 JavaScript,为 Web 开发带来了强大的类型系统、组件化开发支持、与主流框架的无缝集成、大型项目管理能力和提升开发体验等多方面优势。通过明确的类型定义,TypeScript 能够在编码阶段发现潜在错误,提高代码质量;支持组件的清晰定义与复用,增强代码的可维护性;与 React、Vue 等框架结合,提供更佳的开发体验;适用于大型项目,优化代码结构和性能。随着 Web 技术的发展,TypeScript 的应用前景广阔,将继续引领 Web 开发的新趋势。
50 2
|
2月前
|
JavaScript 安全 前端开发
TypeScript类型声明:基础与进阶
通过本文的介绍,我们详细探讨了TypeScript的基础与进阶类型声明。从基本数据类型到复杂的泛型和高级类型,TypeScript提供了丰富的工具来确保代码的类型安全和可维护性。掌握这些类型声明能够帮助开发者编写更加健壮和高效的代码,提高开发效率和代码质量。希望本文能为您在使用TypeScript时提供实用的参考和指导。
48 2
|
2月前
|
JavaScript 开发者
在 Babel 插件中使用 TypeScript 类型
【10月更文挑战第23天】可以在 Babel 插件中更有效地使用 TypeScript 类型,提高插件的开发效率和质量,减少潜在的类型错误。同时,也有助于提升代码的可理解性和可维护性,使插件的功能更易于扩展和升级。
|
3月前
|
JavaScript 前端开发
TypeScript【类型别名、泛型】超简洁教程!再也不用看臭又长的TypeScript文档了!
【10月更文挑战第11天】TypeScript【类型别名、泛型】超简洁教程!再也不用看臭又长的TypeScript文档了!
|
3月前
|
JavaScript 前端开发 安全
TypeScript【基础类型】超简洁教程!再也不用看臭又长的TypeScript文档了!
【10月更文挑战第9天】TypeScript【基础类型】超简洁教程!再也不用看臭又长的TypeScript文档了!
|
3月前
|
JavaScript 前端开发 开发者
深入理解TypeScript:类型系统与最佳实践
【10月更文挑战第8天】深入理解TypeScript:类型系统与最佳实践
|
3月前
|
移动开发 JavaScript 前端开发
TypeScript:数组类型&函数使用&内置对象
本文介绍了 TypeScript 中的数组类型、对象数组、二维数组、函数、函数重载、内置对象等概念,并通过代码示例详细展示了它们的使用方法。还提供了一个使用 HTML5 Canvas 实现的下雨效果的小案例。
|
2月前
|
JavaScript 前端开发 安全
TypeScript进阶:类型系统与高级类型的应用
【10月更文挑战第25天】TypeScript作为JavaScript的超集,其类型系统是其核心特性之一。本文通过代码示例介绍了TypeScript的基本数据类型、联合类型、交叉类型、泛型和条件类型等高级类型的应用。这些特性不仅提高了代码的可读性和可维护性,还帮助开发者构建更健壮的应用程序。
37 0
|
3月前
|
JavaScript 前端开发 开发者
深入理解TypeScript:类型系统与实用技巧
【10月更文挑战第8天】深入理解TypeScript:类型系统与实用技巧