浅谈TypeScript的类型保护机制


Posted in Javascript onFebruary 23, 2020

在编写 TS 时,它做了比我们看到的更多的事情,例如类型保护机制。让我们编写的代码更加严谨,至于怎么回事,让我们来看看吧。

由于这些机制的存在,就算你仍旧以 JS 原生的书写方式,也能帮助你提前发现代码中潜在的问题。(对于认为 TS 语句更复杂的人,也能实现 0 门槛,不改变已有的习惯也能享受静态检测的好处。)

类型保护就是一些表达式,它们会在运行时检查以确保在某个作用域内的类型。

为了更简单的理解,我们首先声明一个联合类型用于举例:

interface Bird {
 fly(): any;
 layEggs(): any;
}

interface Fish {
 swim(): any
 layEggs(): any
}

type Pet = Bird | Fish;

无类型保护时报错

function fn(pet: Pet) { 
 pet.layEggs(); // okay
 pet.swim();  // Error: Property 'swim' does not exist on type 'Bird | Fish'.
}

因为 TS 并不知道 pet 的实例是 Bird 还是 Fish,因此为了谨慎起见,在未手动声明类型时 TS 中只能调用 联合类型 中的 公共方法,例子中未 layEggs() 方法。除非你在调用指定对象数据的属性或方法前,明确告诉 TS 数据对象是一个具体的类型。

类型断言实现类型保护

我需要使用 <Fish>pet 的 类型断言,来告诉 TS 目标对象是什么类型:

function fn(pet: Fish | Bird) {
 if ((<Fish>pet).swim) {
  (<Fish>pet).swim();
 } else {
  (<Bird>pet).fly();
 }
}

虽然这样的断言满足了我们的需求,但并不好方便,需要在各处都进行引用。
备注:如果在编写 tsx 时,你需要将 (<Fish>pet) 写成 (pet as Fish),因为在 tsx 中尖括号 <> 有特殊的含义。

函数中使用 is 定位类型

我们将上面的 if 内的判断封装到函数中,获得更方便的类型保护方式,此函数必须使用 parameterName is Type 的 类型谓语:

function isFish<T>(pet: Fish | Bird): pet is Fish {
 return !!(<Fish>pet).swim;
}

此函数必须返回为 boolean 类型才生效,当返回 true 时则类型定位为 Fish ,返回 false 时则定位为 Fish 之外的类型(多个类型则以 联合类型 定位)。

function fn(pet: Fish | Bird ) { 
 if (isFish(pet)) {
  pet.swim(); // 因 is 语句的生效,此语句块中类型 let pet: Fish
 } else {
  pet.fly(); // 排除 Fish 之外的类型,此语句块中类型 let pet: Bird
 }
}

需要注意是除了 if 中类型生效,TS 还能自动推断出 else 中的类型。

就算你不使用 TS 这些特定的语句,也能享受 类型保护机制 的好处,下面让我们来看看。

使用 typeof 进行类型保护

如果子类型是只是 number、string、boolean、symbol 这几种数据类型,则可以直接使用 typeof 关键字,TS 能够检测并提供类型保护,我们直接引用官方给的例子:

function padLeft(value, padding): string {
  if (typeof padding === "number") {
    return Array(padding + 1).join(" ") + value;  // let padding: number
  }
  else {
    return padding + value;  // let padding: string
  }
}
  
const t1 = padLeft("world", 6);    // "   world"
const t2 = padLeft("world", "hello "); // "hello world"

使用 instanceof 进行类型保护

由于现在前端对于 面向对象 的开发项目越来越多,因此类的引用也更多了。那么类型保护用在此时,可谓是更加有重要,我们使用 instanceof 来达到这一效果:

interface IA { x(): void; }
interface IB { y(): void; }

class A implements IA {
 x() { }
}
class B implements IB {
 y() { }
}

function fn(e: A | B) {
 if (e instanceof A)
  e.x(); // let e: A
 else
  e.y(); // let e: 
}

基于类型保护机制,在语句块中编辑器会给予指定类型的 方法提示,以及类型检测时会提示用户。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
Tab页界面,用jQuery及Ajax技术实现
Sep 21 Javascript
JQuery 操作select标签实现代码
May 14 Javascript
JS、CSS加载中的小问题探讨
Nov 26 Javascript
js如何设置在iframe框架中指定div不显示
Dec 04 Javascript
轻松学习jQuery插件EasyUI EasyUI实现拖动基本操作
Nov 30 Javascript
JS+JSP通过img标签调用实现静态页面访问次数统计的方法
Dec 14 Javascript
Angularjs手动解析表达式($parse)
Oct 12 Javascript
微信小程序 二维码canvas绘制实例详解
Jan 06 Javascript
vue生成随机验证码的示例代码
Sep 29 Javascript
vue.js轮播图组件使用方法详解
Jul 03 Javascript
浅谈JavaScript中this的指向问题
Jul 28 Javascript
React Ant Design树形表格的复杂增删改操作
Nov 02 Javascript
原生javascript制作的拼图游戏实现方法详解
Feb 23 #Javascript
原生javascript运动函数的封装示例【匀速、抛物线、多属性的运动等】
Feb 23 #Javascript
es6中Promise 对象基本功能与用法实例分析
Feb 23 #Javascript
原生JavaScript之es6中Class的用法分析
Feb 23 #Javascript
原生javascript单例模式的应用实例分析
Feb 23 #Javascript
小程序websocket心跳库(websocket-heartbeat-miniprogram)
Feb 23 #Javascript
webpack.DefinePlugin与cross-env区别详解
Feb 23 #Javascript
You might like
smarty+adodb+部分自定义类的php开发模式
2006/12/31 PHP
20个PHP常用类库小结
2011/09/11 PHP
使用PHP遍历文件目录与清除目录中文件的实现详解
2013/06/24 PHP
ThinkPHP中html:list标签用法分析
2016/01/09 PHP
php实现的简单中文验证码功能示例
2017/01/03 PHP
PHP中关于php.ini参数优化详解
2020/02/28 PHP
JavaScript对象、属性、事件手册集合方便查询
2010/07/04 Javascript
JS/jQuery实现默认显示部分文字点击按钮显示全部内容
2013/05/13 Javascript
用js实现table单元格高宽调整,兼容合并单元格(兼容IE6、7、8、FF)实例
2013/06/25 Javascript
给事件响应函数传参数的四种方式小结
2013/12/05 Javascript
JavaScript常用脚本汇总(一)
2015/03/04 Javascript
JavaScript实现基于Cookie的存储类实例
2015/04/10 Javascript
js window对象属性和方法相关资料整理
2015/11/11 Javascript
JavaScript获取当前url根目录(路径)
2016/06/17 Javascript
JQuery页面随滚动条动态加载效果的简单实现(推荐)
2017/02/08 Javascript
jQuery基于随机数解决中午吃什么去哪吃问题示例
2018/12/29 jQuery
vue.js中使用echarts实现数据动态刷新功能
2019/04/16 Javascript
js字符串类型String常用操作实例总结
2019/07/05 Javascript
VUEX-action可以修改state吗
2019/11/19 Javascript
JS判断数组四种实现方法详解
2020/06/29 Javascript
vue的webcamjs集成方式
2020/11/16 Javascript
ssh批量登录并执行命令的python实现代码
2012/05/25 Python
python实现文本进度条 程序进度条 加载进度条 单行刷新功能
2019/07/03 Python
python3实现斐波那契数列(4种方法)
2019/07/15 Python
django的model操作汇整详解
2019/07/26 Python
Python3批量移动指定文件到指定文件夹方法示例
2019/09/02 Python
Python开发企业微信机器人每天定时发消息实例
2020/03/17 Python
记录一下scrapy中settings的一些配置小结
2020/09/28 Python
python集合的新增元素方法整理
2020/12/07 Python
联想阿根廷官方网站:Lenovo Argentina
2019/10/14 全球购物
门诊挂号室室长岗位职责
2013/11/27 职场文书
安全协议书范本
2014/04/21 职场文书
五年级小学生评语
2014/12/26 职场文书
怒海潜将观后感
2015/06/11 职场文书
《棉鞋里的阳光》教学反思
2016/02/20 职场文书
Java实现简易的分词器功能
2021/06/15 Java/Android