软件工程的一个主要部分就是构建组件,构建的组件不仅需要具有明确的定义和统一的接口,同时也需要组件可复用。支持现有的数据类型和将来添加的数据类型的组件为大型软件系统的开发过程提供很好的灵活性。

在C#和Java中,可以使用"泛型"来创建可复用的组件,并且组件可支持多种数据类型。这样便可以让用户根据自己的数据类型来使用组件。

泛型的简单案例

首先,用泛型写一个"Hello World":identity函数。identity函数将会返回我们传入的数据。你可以认为它是个"echo"命令。
不用泛型,我们也不用给identity函数指定类型:

function identity(arg: number): number {return arg;
}

或者,我们可以给identity函数指定"any"类型:

function identity(arg: any): any {return arg;
}

虽然使用"any"类型的时候可以接收任何类型的"arg"参数,但是实际上已经失去函数返回值类型的信息。假如我们传入一个number,我们只知道返回任何类型的值都是可以的。

所以,我们需要一直方式来捕捉参数的类型,也可以用它来表示返回值的类型。这里使用的是"类型变量",一种特殊的变量,代表的是类型而非值。

function identity<T>(arg: T): T {return arg;
}

现在我们已经为identity函数添加了类型变量"T"。"T"允许捕获用户提供的参数类型(如:number),以便我们稍后可以使用该类型。然后我们再次用"T"作为返回值的类型。现在我们可以看到,同一类型被用来作为参数类型和返回值类型。

我们称这个版本的identity函数为泛型,它可用于多种类型。与使用"any"类型不同,它和第一个identity函数(使用number作为参数类型和返回值类型)一样精准(它不会失去任何信息)。

一旦我们定义了泛型函数,有两种方法可以使用。第一种就是传入所有的参数,包括类型参数:

var output = identity<string>("myString"); // output的类型将会是 'string'

在这里,我们明确的将"T"指定为string,作为函数中传入的参数,使用<>包裹该参数而非()。

第二种是最常见的。我们使用/类型推断/,我们希望编译器根据传入的参数自动为"T"指定类型。

var output = identity("myString"); // output的类型将会是 'string'

注意,我们并未显示的给尖括号<>内传入类型,编译器检查"myString",然后将"T"设置为它的类型。虽然类型推断是个很实用的工具,也能够使代码简短易读,但你还是需要跟前面的例子一样明确的传递类型参数,因为可能会存在复杂的函数,使得编译器未能正确的进行类型推断。

使用泛型

当你开始使用泛型,你可能会注意到当你创建一个类似"identity"的泛型函数,编译器会强制要求你在函数中正确的使用这些通用类型参数。也就是说,你真的把这些参数视为可以是任何类型的。

再看看之前的identity函数:

function identity<T>(arg: T): T {return arg;
}

想要每次调用的时候在控制台打印出"arg"参数的length。我们可以尝试这么写:

function loggingIdentity<T>(arg: T): T {console.log(arg.length);  // 错误: T 不存在 .lengthreturn arg;
}

当我们这么做的时候,编译器会抛出一个错误提示我们使用"arg"的".length"属性,但是没有地方指定过"arg"的".length"属性。之前我们说类型变量代表了所有类型,所以可能使用这个函数的时候会传入一个"number"类型的值,而"number"是没".length"属性的。
实际上,我们想让这个函数接受的参数是个"T"类型的数组而非直接"T"。当传入的是数组,length属性便是可用的了。我们可以像创建其他数组类型一样:

function loggingIdentity<T>(arg: T[]): T[] {console.log(arg.length);  // 数组中存在 .length,所以没报错return arg;
}

你可以这样理解loggingIdentity函数:loggingIdentity泛型函数,参数类型是"T",参数"arg"是个类型为"T"的数组,返回的也是个类型为"T"的数组。如果我们传入一个都是数字的数组,那么我们也会得到一个都是数字的数组,因为这时候"T"类型已经绑定为number了。这使得我们可以使用类型变量"T"作为我们使用的类型的一部分,而非全部类型,这也更具灵活性。

我们可以通过这种方式写个例子:

function loggingIdentity<T>(arg: Array<T>): Array<T> {console.log(arg.length);  // 数组中存在 .length,所以没报错return arg;
}

泛型类型

在前面例子中,我们创建了通用的identity函数,可以使用于不同的类型。现在,我们将探讨函数类型及如何创建泛型接口。

泛型函数的类型与非泛型函数一样,只是最前面放上一个类型参数,类似与声明函数:

function identity<T>(arg: T): T {return arg;
}var myIdentity: <T>(arg: T)=>T = identity;

我们也可以给类型中的泛型类型参数指定不同的名称,只要类型变量的数量和其使用方式都能对应的上。

function identity<T>(arg: T): T {return arg;
}var myIdentity: <U>(arg: U)=>U = identity;

我们也可以使用对象字面量的签名调用来写泛型类型:

function identity<T>(arg: T): T {return arg;
}var myIdentity: {<T>(arg: T): T} = identity;

下面开始写第一个泛型接口。用上个例子中的对象字面量来写接口:

interface GenericIdentityFn<T> {(arg: T): T;
}function identity<T>(arg: T): T {return arg;
}var myIdentity: GenericIdentityFn<number> = identity;
var num = myIdentity(10); // 正确,因为类型是number
var str = myIdentity("10") // 错误,参数类型不是number

注意,我们的例子稍微有些改变。我们把非泛型函数签名作为泛型类型的一部分,而不是去描述泛型函数。当我们使用GenericIdentityFn,我们还需要指定对应的类型参数(这里是number),有效的锁定在底层签名调用时会用到的类型。理解"何时将类型参数直接放到签名调用"和"何时将它放到接口上"将有助于描述哪部分类型属于泛型。

除了泛型接口,我们还可以创建泛型类。请注意,不可能创建泛型枚举和模块。

泛型类

泛型类和泛型接口相似。泛型类在类名后面使用尖括号<>包含泛型类型参数列表。

class GenericNumber<T> {zeroValue: T;add: (x: T, y: T) => T;
}var myGenericNumber = new GenericNumber<number>();
myGenericNumber.zeroValue = 1;
myGenericNumber.add = function(x, y) { return x + y; };
alert(myGenericNumber.add(myGenericNumber.zeroValue, 1));  // 2

这是对"GenericNumber"类想当直观的使用,你也可能注意到并未限制只能使用"number"类型。我们可以使用"string"抑或更复杂的对象。

var stringNumeric = new GenericNumber<string>();
stringNumeric.zeroValue = "Hello ";
stringNumeric.add = function(x, y) { return x + y; };alert(stringNumeric.add(stringNumeric.zeroValue, "World")); // Hello World

和接口一样,将类型参数放在类之后来告诉我们类的所有属性都是同一个类型。

正如前面"类"那一节所描述的,一个类由两部分组成:静态部分和实例部分。泛型类仅属于实例部分,所以当我们使用类的时候,静态成员不能使用类的类型参数。

泛型的限制

如果你还记得之前的例子,有时候你想要写一个泛型函数来操作一组类型,并且你是知道这些类型具有什么功能。

在"loggingIdentity"例子中,我们希望能够访问"arg"的".length"属性,但是编译器不能确定每个类型都有".length"属性,所以它将报错提示我们不能这么做。

function loggingIdentity<T>(arg: T): T {console.log(arg.length);  // 错误: T 不存在 .lengthreturn arg;
}

相对于处理任何类型或者所有类型,我们更希望强制去要求函数去处理带有".length"属性的任何类型或者所有类型。只要该类型有这个成员(属性),我们便运行通过,也就是必须包含这个指定的成员(属性)。

既然需要这么做,我们就创建一个描述限制的接口。在这里,先创建一个只有单个属性".length"的接口,然后使用这个接口和"extends"关键字来指明限制:

interface hasLength {length: number;
}function loggingIdentity<T extends hasLength>(arg: T): T {console.log(arg.length);  // 现在我们知道它含有.length属性,并且不报错return arg;
}

因为这个泛型函数现在是有限制的,所以它不在支持任何类型或者所有类型:

loggingIdentity(3); // 错误,number不包含.length属性

因此,我们需要传入其类型具有所需属性的值:

loggingIdentity({length: 10, value: 3}); // 正确

在泛型中使用类类型
当在TypeScript中使用泛型创建工厂函数的时候,需要引用其构造函数的类类型。

class Greeter{greeter:string = "Hello World";
}
function create<T>(c: {new(): T}): T { return new c();
}
var newGreeter = create<Greeter>(Greeter);

TypeScript Generics(泛型)相关推荐

  1. Dart基础之Generics 泛型 <T>

    Java开发过程中,会通过重载实现根据不同的参数类型生成类似方法(多态),进一步可以通过 泛型 进一步提取基类方法:而对于 Dart 而言,为了解决多种方式构造对象的场景,也可以通过 Dart 中的 ...

  2. C# Generics 泛型

    C# Generics 泛型 泛型优点 性能 为了让方法传递任何类型的参数,可以用object类来传递参数.值类型转换为引用类型称为装箱,引用类型转换为值类型称为拆箱,需要强制转换.如下代码所示: 数 ...

  3. typescript学习:Generics泛型

    本文内容如下 泛型Generics的了解与使用 如果你都有了答案,可以忽略本文章,或去TS学习地图寻找更多答案 泛型 定义:占位符,定义时不知道未来需要什么值,等调用的时候再传 作用:提升类,接口,方 ...

  4. React + TypeScript 实现泛型组件

    泛型类型 TypeScript 中,类型(interface, type)是可以声明成泛型的,这很常见. interface Props<T> {content: T; } 这表明 Pro ...

  5. TypeScript 之泛型

    文章出自个人博客 https://knightyun.github.io/2021/04/18/js-ts-generic-type,转载请申明 背景 泛型用于创建可复用的支持多种类型的组件,比如不仅 ...

  6. 一个 TypeScript keyof 泛型用法

    平时工作自认为有 Swift Rust 经验, 所以不需要看 TypeScript 泛型方面的应用, 总以为大同小异, 拒绝看文档学语言, 从我做起. 今日看到一个用上泛型的 pluck 函数 fun ...

  7. JavaSE-Adventure(III): Generics 泛型程序设计

    JavaSE-Adventure (III): 泛型程序设计 CONTENTS JavaSE-Adventure (III): 泛型程序设计 概述 泛型概念 泛型的提出背景 泛型的作用 使用泛型 泛型 ...

  8. 『TypeScript』泛型

    泛型(Generics)是允许同一个函数接受不同类型参数的一种模板.相比于使用 any 类型,使用泛型来创建可复用的组件要更好,因为泛型会保留参数类型. T 代表Type,在定义泛型是通常用作第一个类 ...

  9. 第四章 Generics - 泛型

    现在,你已经知道了Swift中如何实现基本的类和结构,但Swift的强大远不止如此.本章要讲的是Swift的另一个非常强大的语法:泛型. 对于类型安全的语言来说,都有一个常见的问题.想写一个作用于一种 ...

最新文章

  1. Windows 8的圣战
  2. java中文乱码decode_java中文乱码
  3. linux工具之检测内存泄漏-valgrind
  4. 神策数据钟秉哲:一文了解用户标签画像,从洞察到突破
  5. BZOJ - 4196 软件包管理器 (树链剖分+dfs序+线段树)
  6. linux启动过程中内核拷贝,轻松识破linux内核启动过程中的“”套路“”
  7. python支持向量机回归_支持向量机——核函数与支持向量回归(附Python代码)
  8. 九度OJ-1112-导弹拦截-最长不增子序列
  9. mysql连接被拒绝 密码也对_解决Mysql数据库拒绝远程连接和忘记密码的问题
  10. 吴恩达神经网络和深度学习-学习笔记-39-计算机视觉现状
  11. python9_Python9-前端基础知识-day47
  12. Rbf神经网络使用Tensorflow实现
  13. android studio systrace,Android Systrace 基础知识 -- Systrace 简介
  14. Arduino笔记实验(初级阶段)—DHT11温湿度传感器
  15. 使用MATLAB进行多元非线性回归拟合预测
  16. 科目二 离合 要点记录
  17. 人工智能实验二——prolog语言求解渡河问题(传教士和野人渡河,农夫渡河问题)实现详解
  18. hiveserver2 和beeline_Beeline连接Hiveserver2错误
  19. fatal: unable to access ‘https://github.com/NEGVS/the-economist-ebooks.git/‘: schannel: failed to re
  20. csp怎么给线条描边,插画师要失业了?还在纠结阴影怎么画?CSP软件能直接自动生成...

热门文章

  1. 文件重命名批量处理(Matlab)
  2. 2019.01.26 codeforces 1096G. Lucky Tickets(生成函数)
  3. java_day19_MVC和配置文件
  4. rvm RuvyGem Cocoapods brew
  5. 要的需求 ip提取网站源码带采集 要求是PHP源码
  6. Grunt + Bower—前端构建利器
  7. @ViewChild 的三种常用方法
  8. js(Dom+Bom)第八天—Swiper(插件)
  9. 《算法竞赛进阶指南》0.5排序
  10. ES6 数值的扩展