TypeScript Generics(泛型)

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

在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);

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/261858.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Hashtable.ContainsKey跟Hashtable.Contains的区别

最近在用哈希表做项目&#xff0c;发现判断键值的方法有两个 &#xff1a;一个是Contains(object key)&#xff0c;一个是ContainsKey(object key)&#xff0c;就不知道用哪个才对&#xff0c;后来查了资料才知道&#xff0c;其实这两个方法实现的功能是一样的&#xff0c;都是…

html5清除手机页面缓存文件夹,WebView自动缓存-清除缓存

iOS的Webview加载HTML时会自动缓存JS、CSS等文件&#xff0c;当下次加载HTML时会根据请求的缓存策略是否使用缓存本地的JS和CSS&#xff0c;如果本地有缓存&#xff0c;那么直接返回本地资源(判断是否过期)&#xff1b;如果没有本地缓存则向服务器请求地址。1、NSURLRequestCac…

vs2010 快捷键大全

VS2010版快捷键CtrlE,D ----格式化全部代码 CtrlE,F ----格式化选中的代码 CTRL SHIFT B生成解决方案 CTRL F7 生成编译 CTRL O 打开文件 CTRL SHIFT O打开项目 CTRL SHIFT C显示类视图窗口 F4 显示属性窗口 SHIFT F4显示项目属性窗口 CTRL SHIFT E显示资源视图 F12…

swift语言和python区别_Swift为什么能成为编程语言中的黑马?

你好&#xff0c;这里是卖桃者说&#xff0c;今天跟你聊一门年轻又很受欢迎的编程语言&#xff0c;Swift。Swift 是苹果于 2014 年在苹果全球开发者大会(WWDC)上发布的编程语言&#xff0c;它可以与 Objective-C 共同运行于 macOS 和 iOS 平台&#xff0c;用于搭建基于苹果平台…

第一周内容

回到寝室&#xff0c;第一件事情便是想查一下杨老师课上说的“diff”程序。百度了一下&#xff0c;谈到最多的是linux中的diff命令&#xff0c;于是换个关键词&#xff0c;diff工具&#xff0c;发现diffmerge&#xff0c;P4merge&#xff0c;kdiff&#xff0c;TextDiff等等。以…

osgdem的参数表(转)

osgdem的参数表 ParameterDescriptionDefault General -h or --help Display commandline arguments information --task -s Specify a VPB source file containing all commandline options. --so Output the VPB source file for the current run. --report --cache Read the…

html中如何把两行合并单元格,css合并两列单元格内容

用纯DIVCSS做一个两行两列的表格&#xff0c;但第二列中两行怎么在html里把一行中的两列合并世界最不可以相信的话,就是从女人嘴里说出的话〃如上图&#xff0c;怎么做出上面图中的效果&#xff0c;分享大神详解CSS表格单元格占两行可以参考以下的代码&#xff1a; 单元格占两行…

ASP.NET MVC 实现二级域名(泛域名)

自从微软发布 ASP.NET MVC 和routing engine (System.Web.Routing)以来&#xff0c;就设法让我们明白你完全能控制URL和routing&#xff0c;只要与你的application path相结合进行扩展&#xff0c;任何问题都迎刃而解。如果你需要在所处的域或者子域处理数据标记的话&#xff0…

list和tuple

2019独角兽企业重金招聘Python工程师标准>>> list Python内置的一种数据类型是列表&#xff1a;list。list是一种有序的集合&#xff0c;可以随时添加和删除其中的元素。 比如&#xff0c;列出班里所有同学的名字&#xff0c;就可以用一个list表示&#xff1a; >…

springboot数据源不正确_Spring MVC 到 Spring Boot 的简化之路

Spring全家桶笔记&#xff1a;SpringSpring BootSpring CloudSpring MVC​shimo.im01 背景从Servlet技术到Spring和Spring MVC&#xff0c;开发Web应用变得越来越简捷。但是Spring和Spring MVC的众多配置有时却让人望而却步&#xff0c;相信有过Spring MVC开发经验的朋友能深刻…

KDT#94 为DW/BI系统建立定制工具

Building Custom Tools for the DW/BI System 市场上有大量的工具帮我们来建立DW/BI系统、把信息交付给业务用户。这些工具的种类也很多&#xff0c;它们包括关系型数据库管理系统、OLAP数据库管理系统、ETL工具、数据挖掘工具、查询工具、报表工具&#xff0c;以及BI门户工具等…

c盘所有的html文件全删,我将C盘文件夹全删了

如果删除了系统文件系统运行就会出现错误&#xff0c;如果系统运行稳定就说明没有删除系统重要文件&#xff0c;建议今后删除文件时尽量弄清楚后在删除&#xff0c;如果系统出现问题&#xff0c;可以按照下面方法修复。1、请你用系统自带的系统还原&#xff0c;还原到你没有出现…

OSSIM主要数据库表结构

OSSIM主要数据库表结构对于从事OSSIM开发的技术人员&#xff0c;最主要的需要知道OSSIM库里的多种表结构&#xff0c;下面举几个典型事例&#xff1a;/* config表 */DROP TABLE IF EXISTS conf;CREATE TABLE conf (recovery int NOT NULL,threshold int NOT NUL…

C#线程同步(1)- 临界区&Lock .

预备知识&#xff1a;线程的相关概念和知识&#xff0c;有多线程编码的初步经验。 一个机会&#xff0c;索性把线程同步的问题在C#里面的东西都粗略看了下。 第一印象&#xff0c;C#关于线程同步的东西好多&#xff0c;保持了C#一贯的大杂烩和四不象风格(Java/Delphi)。临界区跟…

python atm银行取款系统_Python实现ATM系统

今天偶尔在知乎上看到某大佬用Python写的ATM系统案例&#xff0c;然后观摩了下他的实现思路和源码&#xff0c;感觉受益颇多&#xff0c;于是就根据自己的思路和目前掌握的Python编程基础将ATM实现了一下&#xff0c;以下是案例解析的过程&#xff1a;案例剖析&#xff1a;1.at…

【摘录】BREW应用的c++实现注意点

BREW应用的c实现注意点 从VC6.0的调试器来说吧&#xff0c;肯定支持C语言了&#xff0c;对于ARM或者GCC来说&#xff0c;也是有可用的C编译器&#xff0c;而且任何支持BREW的手机都可以运行通过ARM或者GCC编译连接出来的目标代码&#xff0c;所以从环境来说&#xff0c;BREW开发…

MapXtreme 包含所有自带坐标系一览

CoordSys 对象包含关于 X 和 Y 坐标如何与其在 Earth 上的位置相关联的基本信息。 每个 Geometry 或 Map 对象都有一个关联的坐标系。 CoordSys 对象包含对坐标系的详细说明。 CoordSysFactory 类提供了各种用于创建不同 CoordSys 对象的方法。 所有 CoordSys 对象都是只读的&a…

html4的语法,HTML——语法

文章目录页面结构一个标准的HTML页面如下&#xff1a;Document一、文档声明头标准的HTML页面&#xff0c;第一行是以开头的的语句&#xff0c;这就是文档声明头&#xff0c;即DocType Declaration&#xff0c;简称DTD。DTD可以告知浏览器使用哪种HTML或者XHTML规范。二、页面语…

如何学习streamdecoder类_2019年终巨献:一份拿下了阿里、网易、滴滴等大厂offer的学习笔记...

2019仅剩最后二十天&#xff0c;回顾今年初遇“寒冬”时&#xff0c;自己也挺慌的&#xff0c;但是经历过这么多次面试后&#xff0c;我才“醒悟”&#xff0c;所谓的“寒冬”&#xff0c;“冻死”的都是“衣服穿的少的”。年末了在这里做一个年度总结&#xff0c;今年面试了不…

自定义控件的构建(12)

Share 前面讲了模板的构建&#xff0c;我们忽略了一个细节&#xff0c;如果接触ASP.NET时间不长的话&#xff0c;一般都会看到数据表达式是<%#Eval(‘Name’)%>这种形式的&#xff0c; 那么我们为什么用<%#Container.Name%>这种形式呢&#xff0c;其实前者是ASP.NE…