TypeScript魔法堂:函数类型声明其实很复杂
前言
江湖有传“动态类型一时爽,代码重构火葬场”,由于动态类型语言在开发时不受数据类型的约束,因此非常适合在项目原型阶段和初期进行快速迭代开发使用,这意味着项目未来将通过重写而非重构的方式进入成熟阶段。而在企业级应用开发中,每个系统特性其实都是需求分析人员与用户进行多次调研后明确下来的,后期需要重写的可能性微乎其微,更多的是修修改改,在单元测试不足常态化的环境下静态类型的优势就尤为突出。而TypeScript的类型系统和编译时类型检查机制则非常适合用于构建企业级或不以重写实现迭代升级的应用系通。
本系列将重点分享TypeScript类型声明相关实践
- 函数类型声明其实很复杂
- 玩转交叉类型和联合类型
- class,inteface和type到底选哪个?
- 从lib.d.ts学习外部类型声明的最佳实践
- 类型声明综合实战
本文为该系列的首发,那么我们现在就开始吧!
定义即声明
当我们通过TypeScript定义函数时,实际上已经声明了函数签名和定义了函数体。
function foo(message: string, count?: number, displayLog = true): never {
console[displayByLog ? 'log' : 'warn'](`message: ${message}; count: ${count}`)
throw new Error('Just a error.')
}
上述函数定义附带声明了function foo(x: boolean, y: string, z: undefined | number): never函数签名,这里我特意替换参数名称以便大家将关注点放在函数参数列表类型和返回值类型上。
后续通过如下代码调用foo函数
foo('hi') // 回显 message: hi; count: undefined
foo('hi', 'yes') // 编译报错
函数重载
JavaScript中我们会通过函数重载来整合处理入参数据结构存在差异,但处理意图和处理结果相同的行为。具体实现方式有
function querySelector(x, parent) {
var arg1 = typeof x === 'string' ? 0 : 1
var arg2 = parent instanceof HTMLElement ? 0 : 1
return (querySelector.overloads[arg1][arg2]).call(null, x, parent)
}
function q00 (x /*: string*/, p /*: HTMLElement*/) {
return p.querySelector(x)
}
function q01 (x /*: string*/, p /*: JQuery*/) {
return p.find(x)[0]
}
function q10 (x /*: JQuery*/, p /*: HTMLElement*/) {
return $(p).find(x)[0]
}
function q11 (x /*: JQuery*/, p /*: JQuery*/) {
return p.find(x)[0]
}
querySelector.overloads = [[q00,q01],[q10,q11]]
而TypeScript中的函数重载并没有让我们定义得更轻松,可以理解为在原JavaScript实现的基础上添加类型声明信息,这样反而让定义变得复杂,但为了能更安全地调用却是值得的。
写法1:
function querySelector(x: string, p: HTMLElement): HTMLElement
function querySelector(x: string, p: JQuery): HTMLElement
function querySelector(x: JQuery, p: HTMLElement): HTMLElement
function querySelector(x: JQuery, p: JQuery): HTMLElement
// 和JavaScript一样需要定义一个Dispatch函数,用于实现调用重载函数的具体规则
function querySelector(x, y) {
var arg1 = typeof x === 'string' ? 0 : 1
var arg2 = parent instanceof HTMLElement ? 0 : 1
if (arg1 === 0 && arg2 === 0) {
return p.querySelector(x)
}
else if (arg1 === 0 && arg2 === 1) {
return p.find(x)[0]
}
else if (arg1 === 1 && arg2 === 0) {
return $(p).find(x)[0]
}
else {
return p.find(x)[0]
}
}
写法2:
interface QuerySelector{
(x: string, p: HTMLElement): HTMLElement
(x: string, p: number): HTMLElement
(x: number, p: HTMLElement): HTMLElement
(x: number, p: number): HTMLElement
overloads: Function[][]
}
// 和JavaScript一样需要定义一个Dispatch函数,用于实现调用重载函数的具体规则
let querySelector: QuerySelector= <QuerySelector>function (x: string | number, p: HTMLElement | number): HTMLElement {
let arg1 = typeof x === 'string' ? 0 : 1
let arg2 = parent instanceof HTMLElement ? 0 : 1
return (querySelector.overloads[arg1][arg2]).call(null, x, parent)
}
function q00 (x: string, p: HTMLElement):HTMLElement {
return p.querySelector(x)
}
function q01 (x: string, p: JQuery):HTMLElement {
return p.find(x)[0]
}
function q10 (x: JQuery, p: HTMLElement):HTMLElement {
return p.find(x)[0]
}
function q11 (x: JQuery, p: JQuery):HTMLElement {
return p.find(x)[0]
}
querySelector.overloads = [[q00, q01],[q10, q11]]
写法2注意事项:
- Dispatch函数必须采用
<T>作为类型断言而不能使用as进行类型转; - Dispatch函数必须通过
function方式定义,而不能使用箭头函数方式定义。
如果想以箭头函数的方式定义Dispatch函数,那么写法就会更复杂了。
interface QuerySelector{
(x: string, p: HTMLElement): HTMLElement
(x: string, p: number): HTMLElement
(x: number, p: HTMLElement): HTMLElement
(x: number, p: number): HTMLElement
}
interface Overload {
overloads: Function[][]
}
let querySelector: <QuerySelector & Overload>
let querySelectorDispatch:<QuerySelector> = (x: string | number, p: HTMLElement | number): HTMLElement => {
let arg1 = typeof x === 'string' ? 0 : 1
let arg2 = parent instanceof HTMLElement ? 0 : 1
return (querySelector.overloads[arg1][arg2]).call(null, x, parent)
}
function q00 (x: string, p: HTMLElement):HTMLElement {
return p.querySelector(x)
}
function q01 (x: string, p: JQuery):HTMLElement {
return p.find(x)[0]
}
function q10 (x: JQuery, p: HTMLElement):HTMLElement {
return p.find(x)[0]
}
function q11 (x: JQuery, p: JQuery):HTMLElement {
return p.find(x)[0]
}
querySelector = querySelectorDispatch as QuerySelector & Overload
querySelector.overloads = [[q00, q01],[q10, q11]]
累死人了。。。。。。。
高阶函数的类型声明
高阶函数作为JavaScript最为人称道的特性,在TypeScript中怎能缺席呢?
// 1
let foo1: (message: string, count?: number, displayLog?: boolean) => never
// 2
interface FooDecl {
(message: string, count?: number, displayLog?: boolean): never
}
let foo2: FooDecl
// 3
let foo3: {(message: string, count?: number, displayLog?: boolean): never}
// 4
type FooType = (message: string, count?: number, displayLog?: boolean) => never
上述为4种声明高阶函数类型的写法,其中第3种是第2种的简写形式。
1、2和3方式声明了变量的值类型,而2中的interface FooDecl和4中则声明类型本身。
foo1,foo2,foo3作为变量(value)可作为传递给函数的实参,和函数的返回值。因此针对它们的值类型声明是无法被重用的,也无法用于函数声明和其它类型声明中;
FooDecl,FooType作为类型声明,及可以被反复重用在各函数声明和其它类型声明中。
函数类型兼容
函数类型兼容的条件:
- 形参列表个数小于等于目标函数类型的形参列表个数;
- 形参列表中形参类型的顺序和目标函数类型的形参列表一致,或形参类型为目标函数类型相应位置的参数类型的子类型;
- 函数返回值必须为目标函数类型返回值的子类型。
const add: (x: number, y: number) => number = (x, y) => x + y
const increment(x: number) => number = x => x+1
add = increment // 类型兼容
increment = add // 类型不兼容
const handleEvent: (e: Event) => void;
const handleMouseEvent: (e: MouseEvent) => void;
handleEvent = handleMouseEvent // 类型兼容
handleMouseEvent = handleEvent // 类型不兼容
总结
函数类型声明难点在于函数重载这一块,而作为库开发者函数重载往往能帮助我们开发出更容易记忆使用和优雅的接口,既然逃不过那不如好好努力克服困难吧!
转载请注明来自:https://www.cnblogs.com/fsjohnhuang/p/13903589.html —— ^_^肥仔John
TypeScript魔法堂:函数类型声明其实很复杂的更多相关文章
- TypeScript基础类型,类实例和函数类型声明
TypeScript(TS)是微软研发的编程语言,是JavaScript的超集,也就是在JavaScript的基础上添加了一些特性.其中之一就是类型声明. 一.基础类型 TS的基础类型有 Boolea ...
- TypeScript魔法堂:枚举的超实用手册
前言 也许前端的同学会问JavaScript从诞生至今都没有枚举类型,我们不是都活得挺好的吗?为什么TypeScript需要引入枚举类型呢? 也许被迫写前端的后端同学会问,TypeScript的枚举类 ...
- TypeScript入门三:TypeScript函数类型
TypeScript函数类型 TypeScript函数的参数 TypeScript函数的this与箭头函数 TypeScript函数重载 一.TypeScript函数类型 在上一篇博客中已经对声明Ty ...
- TypeScript - 类型声明、枚举、函数、接口
目录 可定义的类型 类型声明 枚举 函数 接口 可定义的类型 以下所写的并不代表typescript的数据类型,而是在使用过程中可以用作定义的类型 number : 数值类型: string ...
- JS魔法堂:那些困扰你的DOM集合类型
一.前言 大家先看看下面的js,猜猜结果会怎样吧! 可选答案: ①. 获取id属性值为id的节点元素 ②. 抛namedItem is undefined的异常 var nodes = documen ...
- TypeScript 函数-函数类型
//指定参数类型 function add(x:number,y:number){ console.log("x:"+x); // reutrn(x+y); } //指定函数类型 ...
- typescript函数类型接口
/* 接口的作用:在面向对象的编程中,接口是一种规范的定义,它定义了行为和动作的规范,在程序设计里面,接口起到一种限制和规范的作用.接口定义了某一批类所需要遵守的规范,接口不关心这些类的内部状态数据, ...
- WebComponent魔法堂:深究Custom Element 之 标准构建
前言 通过<WebComponent魔法堂:深究Custom Element 之 面向痛点编程>,我们明白到其实Custom Element并不是什么新东西,我们甚至可以在IE5.5上定 ...
- JS魔法堂:jsDeferred源码剖析
一.前言 最近在研究Promises/A+规范及实现,而Promise/A+规范的制定则很大程度地参考了由日本geek cho45发起的jsDeferred项目(<JavaScript框架设计& ...
随机推荐
- 线上Redis高并发性能调优实践
项目背景 最近,做一个按优先级和时间先后排队的需求.用 Redis 的 sorted set 做排队队列. 主要使用的 Redis 命令有, zadd, zcount, zscore, zrange ...
- 033 01 Android 零基础入门 01 Java基础语法 03 Java运算符 13 运算符和表达式知识点总结
033 01 Android 零基础入门 01 Java基础语法 03 Java运算符 13 运算符和表达式知识点总结 本文知识点:运算符和表达式知识点总结 前面学习的几篇文都是运算符和表达式相关的知 ...
- [源码阅读] 阿里SOFA服务注册中心MetaServer(2)
[源码阅读] 阿里SOFA服务注册中心MetaServer(2) 目录 [源码阅读] 阿里SOFA服务注册中心MetaServer(2) 0x00 摘要 0x01 MetaServer 注册 1.1 ...
- Java学习之动态代理篇
Java学习之动态代理篇 0x00 前言 在后面的漏洞研究的学习中,必须要会的几个知识点.反射机制和动态代理机制.至于反射的前面已经讲到过了,这里就不做更多的赘述了. 0x01 动态代理 这里先来讲一 ...
- EDI模拟实验
EDI模拟实验 [实验目的] ⑴.了解EDI报文的格式和特点. ⑵.掌握EDI报文生成和发送流程. [实验条件] ⑴.个人计算机一台,预装Windows XP操作系统和浏览器 ⑵.计算机通过局域网形式 ...
- linux centos 04
1.python的虚拟环境 1.将当前机器上的解释器作为一个 本地,复制出的很多歌 虚拟解释器 物理机上的 本体解释器 ,什么事也不做 分身1: 解释器1:虚拟环境1 运行django 1 ...
- unix socket接口
socket 创建套接字文件: #include <sys/socket.h> // 成功返回非负套接字描述符,失败返回-1 int socket(int domain, int type ...
- web自动化测试总结
web自动化: 1.测试用例(操作步骤,熟读需求文档,web项目先用手工研究,前置条件,预期结果) 接口自动化测试中数据功能最适合作为数据驱动,数据放在excel中需要操作excel 为什么web自动 ...
- 【C语言入门学习笔记】如何把C语言程序变成可执行文件!
环境 在ANSI的任何一种实现中,存在两种不同的环境. 翻译环境:在这个环境里,源代码被转换为可执行的机器指令. 执行环境:用于实际执行代码. 翻译环境 组成一个程序的每个源文件通过编译过程分别转成目 ...
- spring cloud:通过client访问consul集群(spring cloud hoxton sr8 / spring boot 2.3.4)
一,为什么要搭建consul的client? 1,网上的很多资料,访问consul时用的单机模式,这样是不可以直接在生产环境中使用的 还有一些资料,搭建了consul的集群后,直接访问集群中的某一个i ...