前言
TypeScript真香系列的内容将参考中文文档,但是文中的例子基本不会和文档中的例子重复,对于一些地方也会深入研究。另外,文中一些例子的结果都是在代码没有错误后编译为JavaScript得到的。如果想实际看看TypeScript编译为JavaScript的代码,可以访问TypeScript的在线编译地址,动手操作,印象更加深刻。
概念
TypeScript的泛型与其他面向对象的语言中的定义是相似的。泛型可以理解为在我们定义函数、接口或者类的时候,不预先指定其相关的类型,而是在使用的时候手动指定类型。这和TypeScript基础类型中的any是有区别的。
当我们不使用泛型的时候,可能就如下这样:
function show(name: string): string {
return name;
}
function show(name: any): any {
return name;
}
上面例子中用any或许没有什么问题,但是当我们想传入的参数类型和返回值类型相同时,使用any时的返回值就可能有多种情况了。还有就是我们平时也会考虑重用性,提取公共组件等。这时泛型就能派上了。下面是一个泛型函数的例子:
function show<T>(arg: T): T {
return arg;
}
这个show函数后面的 T ,叫做类型变量,这个我们后面会介绍。这个函数我们可以用两种方法调用。
第一种为传入参数,包括类型参数:
show<string>("james"); //"james"
第二种为不包括类型参数,这种方式利用了类型推论,这个再后面的章节会讲到:
show(1); //1
泛型变量
上面所写的例子中,我们可以在“<>”中指定一个变量T,并且我们要把这个T定义在参数和返回值上。编译器在检查我们的代码的时候,我们必须在函数体中正确的使用这个通用的类型。
function show<T>(arg: T): T {
console.log(arg.length) //错误,类型T不存在length属性
return arg;
}
上面的例子中,我们在打印arg的length中报错了。原因是我们在使用这个函数的时候,传入的参数可能为数字,而数字是没有length属性的。
当然,我们在调用函数的时候也要正确调用:
function show<T>(arg: T): T {
return arg;
}
show<string>(1); //错误,参数1的类型不是string
show<>(1); //错误,参数类型不能为空
针对数组:
function show<T>(arg: T[]): T[] {
return arg;
}
show<string>(["奥","利","给"]); //["奥", "利", "给"]
function show<T>(arg: Array<T>): Array<T> {
return arg;
}
show<number>([6, 6, 6]); // [6, 6, 6]
泛型接口
如下所示:
interface IMan {
<T>(arg: T): T
}
function man<T>(arg: T): T{
return arg;
}
let showMan: IMan = man;
showMan("james"); //"james"
还有另外一种写法,这种方法我们可以把泛型参数当做整个接口的参数:
interface IMan<T> {
(arg: T): T
}
function man<T>(arg: T): T{
return arg;
}
let showMan: IMan<string> = man;
showMan("james"); //"james"
这样的好处就是在使用时知道具体的参数类型了。
泛型类
我们可以使用(<>)和尖括号中间的变量,然后跟在类名的后面来表示泛型类。需要注意的是,泛型类指的是实例部分的类型,类的静态属性不能使用泛型类型。
class Man<T> {
name: T;
constructor(arg: T) {
this.name = arg;
}
play(a: T) : T {
return a;
}
}
let showMan = new Man<string>("James");
showMan.name; //"James"
showMan.play("ball"); //"ball"
也可以稍微复杂一点:
class Man<T> {
name: T;
sum: T[];
constructor(arg: T) {
this.name = arg;
this.sum = [];
}
push(item: T): void{
this.sum.push(item)
}
output() {
return this.sum;
}
}
let showMan = new Man<string>("James");
showMan.push("one for all");
showMan.output(); //["one for all"]
泛型约束
让我们回到泛型变量一节中的一个例子:
function show<T>(arg: T): T {
console.log(arg.length) //错误,类型T不存在length属性
return arg;
}
如果我们想要使用length属性,那该怎么办?可以看下面的例子:
interface ILen {
length: number;
}
function show<T extends ILen>(arg: T): T {
console.log(arg.length) //没有报错了
return arg;
}
虽然上面的例子没有再报错,但是因为这个泛型函数被定义了约束,所以我们再调用函数的时候又有了讲究,也就是我们在传值的时候需要像下面的例子一样:
interface ILen {
length: number;
}
function show<T extends ILen>(arg: T): T {
console.log(arg.length)
return arg;
}
show(3); //错误,参数3不能分配给类型ILen
show({ length: 2 }); // 2 {length: 2}
show({length:2, arg:1}) //2 {length: 2, arg: 1}
show({length:2, arg:"james"}) //2 {length: 2, arg: "james"}
参考
https://github.com/zhongsp/Ty...
最后
文中有些地方可能会加入一些自己的理解,若有不准确或错误的地方,欢迎指出~
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。