Typescript第七章 处理错误(返回null,抛出异常,返回异常,Option类型)
第七章 处理错误
Typescript竭尽所能,把运行时异常转移到编译时。Typescript是功能丰富的系统,加上强大的静态和符号分析能力,包揽了大量辛苦的工作。
但是有些问题是无法避免的,比如网络和文件系统异常,解析用户输入时出现额错误,堆栈溢出及内存不足。不过,Typescript的类型系统足够强大,提供了很多处理运行时错误的方式,不会眼看程序崩溃。
- 返回null
- 抛出异常
- 返回异常
- Option类型
7.1 返回null
function ask() {return prompt("when is your birthday")}function isValid(data: Date) {return Object.prototype.toString.call(date) === '[object Date]' &&!Number.isNaN(date?.getTime())}function parse(birthday: string | null): Date | null {let date;// null判断if (birthday) {date = new Date(birthday)if (!isValid(date)) {return null}}return new Date}let date = parse(ask())// 强制判断是否为nullif (date) {console.info("Date is ", date.toISOString());} else {console.error("Error parsing date for some reason");}
返回null是处理错误最轻量的方式。
然而这么做丢失了一些信息,调用方不知道为什么错误。
7.2 抛出异常
把返回null改成抛出异常
function parse(birthday:string):Date{let date = new Date(birthday)if(!isValid(date)){throw new RangeError("enter a date in the form YYYY/MM/DD")}return date}//使用时捕获错误try{let date = parse(ask())log(date.toISOString())}catch(e){if(e instanceof RangeError){log(e.message)}else{throw e}}
自定义错误类型
class InvalidDateFormatError extends RangeError{}class DateIsInTheFutureError extends RangeError{}function parse(birthday:string):Date{let date = new Date(birthday)if(!isValid(date)){throw new InvalidDateFormatError("enter a date in the form YYYY/MM/DD")}if(date.getTime()>Date.now()){throw new DateIsInTheFutureError("are you a timelord")}return date}
7.3 返回异常
Typescript不支持throws子句。不过我们可以使用并集类型近似实现这个特性
throws子句的作用是指出一个方法可能抛出什么运行时异常,让使用方知道该处理那些异常
class InvalidDateFormatError extends RangeError { }class DateIsInTheFutureError extends RangeError { }/**@thorws{InvalidDateFormateError} xxxx**/function parse(birthday: string|null): Date | InvalidDateFormatError | DateIsInTheFutureError {let date = new Date(birthday!)if (!isValid(date)) {return new InvalidDateFormatError("enter a date in the form YYYY/MM/DD")}if (date.getTime() > Date.now()) {return new DateIsInTheFutureError("are you a timelord")}return date}let result = parse(ask())// 手动判断可能出现的异常if(result instanceof InvalidDateFormatError){console.log("");}else if(result instanceof DateIsInTheFutureError){console.log("----");}else{console.log("ok");}
这里,我们利用Typescript的类型系统实现了
- 在parse函数的签名中加入可能出现的异常
- 告诉使用方可能抛出那些异常
- 迫使使用方处理(或再次抛出)每一个异常
使用方太烂的话,可以不用逐一处理各个异常,但是要明确写出来
if(result instanceof Error){log}else{log}
这种方式的缺点是,串联和嵌套可能会让人觉得烦躁。如果一个函数返回T|Error1,那么使用该函数的函数有两个选择:
- 显示处理Error1
- 处理T(成功的情况),把Error1传给使用方处理。这样传来传去,使用方要处理的错误将越来越多。
function x():T|Error1{}function y():U|Error1|Error2{let a = x()if(a instanceof Error){return a}}function z():U|Error1|Error2|Error3{let a = y()if(a instanceof Error){return a}}
7.4 Option类型
除此之外,还可以使用专门的数据类型描述异常。这种方式与返回值和错误的并集相比是有缺点的(尤其是与不使用这些数据类型的代码互操作时),但是却便于串联可能出错的操作。在这方面,常用的三个选项是Try(try type),Option(也叫Maybe)和Either(either type)类型。本章只介绍Option类型,其他两个类型本质上基本相同。
注意:Try,Option,和Either与Array,Error,Map或Promise不同,不是JavaScript环境内置的数据类型。如果想使用,要在NPM中寻找实现,或者自己编辑
Option类型源自Haskell,OCaml,Scala和Rush等语言,隐含的意思是,不返回一个值,而是返回一个容器,该容器里可能有一个值,也可能没有。 这个容器有一些方法,即使没有值也能串联操作。容器几乎可以是任何数据结构,只要能在里面存放值。例如,可以使用数组作为容器:
function parse(birthday: string | null): Date[] {if (birthday) {let date = new Date(birthday)if (!isValid(date)) {return []}return [date]}return []}function isValid(date: Date) {return Object.prototype.toString.call(date) === '[object Date]' &&!Number.isNaN(date?.getTime())}let ask = (): string | null => {let number = Math.random();if (number > 0.5) {return "2023/08/01"} else {return null}}let date = parse(ask())date.map(_=>_.toISOString()).forEach(_=>console.log("date is",_))
你可能注意到了,Option的缺点与返回null一样,只告诉使用方什么地方出错了,而未说明出错的原因。
Option真正发挥作用是在一次执行多个操作,而每个操作都有可能出错。
例如,我们之前一定假定ask一定成功,而parse可能失败。但是如果ask失败了,我们不能放置不理,继续计算,此时,仍然可以使用Option。
let ask = () => {let result = "2023/08/01"if (result === null) {return []}return [result]}ask().map(parse).map(date=>date.toISOString())// 报错.forEach(date=>console.log("date is",date))
出错了。这是因为我们把一个Date数组(Date[])映射到了一个Date数组构成的数组上(Date[][]
),解决方法是在操作之前整平数组:
let ask = () => {let result = "2023/08/01"if (result === null) {return []}return [result]}flatten(ask().map(parse)).map(date=>date.toISOString())// 报错.forEach(date=>console.log("date is",date))function flatten<T>(array:T[][]):T[]{return Array.prototype.concat.apply([],array)}
这样有点不便,Option类型没有告诉我们什么信息(一切都是常规数组)。为了改变这种局面,我们可以把整个过程(把一个值放入一个容器,提供操作那个值的方式,提供从容器中取回结果的方式)包装成一个特殊的数据结构,让一切一目了然。实现好以后,可以像下面这样使用
ask().flatMap(parse).flatMap(date => new Some(date.toISOString())).flatMap(date => new Some("date si "+date)).getOrElse("Error parsing date for some reason")
我们将才用下述方式定义Option类型:
- Option是一个接口,实现两个类:Some和None,这是两个Optioin。Some是包含一个T类型值的Option,None是没有值的Option,表示失败。
- Option即是类型也是函数。作为类型,他是一个接口,表示Some和None的超类型。作为函数,他是创建Option类型值的方式。
interface Option<T>{} //1.class Some<T> implements Option<T> {//2.constructor(private value:T){}}class None implements Option<never>{}//3
- Option是一个接口,Some和None都可以实现该接口
- Some表示操作成功,得到一个值。与前面使用的数组一样,Some是改值的容器。
- None表示失败,不包括值。
这几个类型等效于下述通过数组实现的Option:
- Option是[T]|[]
- Some是[T]
- None是[]
我们能对Option做些什么呢,就目前
flatMap
串联操作可能为空的Option
getOrElse
从Option取得值
首先在Option接口中定义这两个操作,然后在Some和None中具体实现:
interface Options<T> {flatMap<U>(f: (value: T) => None): NoneflatMap<U>(f: (value: T) => Options<U>): Options<U>getOrElse(value: T): T;}class Some<T> implements Options<T> {constructor(private value: T) {}flatMap<U>(f: (value: T) => None): NoneflatMap<U>(f: (value: T) => Some<U>): Some<U>flatMap<U>(f: (value: T) => Options<U>): Options<U> {return f(this.value);}getOrElse(): T {return this.value;}}class None implements Options<never> {flatMap(): None {return this;}getOrElse<U>(value: U): U {return value;}}function Options<T>(value:null|undefined):Nonefunction Options<T>(value:T):Some<T>function Options<T>(value:T):Options<T>{if(value == null){return new None}return new Some(value)}
ask().flatMap(parse).flatMap(date => new Some(date.toISOString()))// Options<string>.flatMap(date => new Some("date si " + date))// Options<string>.getOrElse("Error parsing date for some reason")// string// let result = Options(6)// .flatMap(n=>Options(n*3))// .flatMap(n=>new None)// .getOrElse(7)
Option也不是没有缺点,Option通过一个None表示失败,没有关于失败的详细信息,也不知道失败的原因。另外与不使用Option的代码无法互操作(要自己手动包装API,让他们返回Option)
相关文章:

Typescript第七章 处理错误(返回null,抛出异常,返回异常,Option类型)
第七章 处理错误 Typescript竭尽所能,把运行时异常转移到编译时。Typescript是功能丰富的系统,加上强大的静态和符号分析能力,包揽了大量辛苦的工作。 但是有些问题是无法避免的,比如网络和文件系统异常,解析用户输入…...

Qt库xcb问题
首先在~/.bashrc中加入 export QT_DEBUG_PLUGINS1然后看具体的报错 查看某个库链接的库: ldd libqxcb.so然后找到真正缺少的库,再在路径下搜索,然后建立软链接。 https://blog.csdn.net/LOVEmy134611/article/details/107212845 https://…...

C++ | 哈希表的实现与unordered_set/unordered_map的封装
目录 前言 一、哈希 1、哈希的概念 2、哈希函数 (1)直接定址法 (2)除留余数法 (3)平方取中法(了解) (4)随机数法(了解) 3、哈…...

【漏洞挖掘】Xray+rad自动化批量漏洞挖掘
文章目录 前言一、挖掘方法二、使用步骤工具安装使用方法开始挖掘 总结 前言 自动化漏洞挖掘是指利用计算机程序和工具来扫描、分析和检测应用程序、网络和系统中的安全漏洞的过程。这种方法可以帮助安全专家和研究人员更高效地发现和修复潜在的安全威胁,从而提高整…...

Swagger UI教程 API 文档和Node的使用
在团队开发中,一个好的 API 文档可以减少很多交流成本,也可以使一个新人快速上手业务。 前言 swagger ui是一个API在线文档生成和测试的利器,目前发现最好用的。为什么好用?Demo 传送门 支持API自动生成同步的在线文档 这些文档可…...

P5691 [NOI2001] 方程的解数
[NOI2001] 方程的解数 题目描述 已知一个 n n n 元高次方程: ∑ i 1 n k i x i p i 0 \sum\limits_{i1}^n k_ix_i^{p_i} 0 i1∑nkixipi0 其中: x 1 , x 2 , … , x n x_1, x_2, \dots ,x_n x1,x2,…,xn 是未知数, k 1 ,…...

rust里用什么表示字节类型?
在Rust中,字节可以使用 u8 类型来表示。 u8 是一个无符号8位整数类型,可以表示0到255之间的值,对应于一个字节的范围。 以下是一个示例,演示了如何声明和使用字节: fn main() {let byte: u8 65; // 表示字母A的ASCI…...

CMake简介
文章目录 为什么需要头文件为什么 C 需要声明头文件 - 批量插入几行代码的硬核方式头文件进阶 - 递归地使用头文件 CMake什么是编译器多文件编译与链接CMake 的命令行调用为什么需要库(library)CMake 中的静态库与动态库CMake 中的子模块子模块的头文件如…...

[threejs]相机与坐标
搞清相机和坐标的关系在threejs初期很重要,否则有可能会出现写了代码,运行时一片漆黑的现象,这种情况就有可能是因为你相机没弄对。 先来看一下threejs中的坐标(世界坐标) 坐标轴好理解,大家只需要知道在three中不同颜色代表的轴…...

Qt信号与槽机制的基石-MOC详解
引入 上篇讲到了信号与槽就是实现的观察者模式,那具体如何生成映射表就是moc做的事情。 一、moc简介 1. moc的定义 moc 全称是 Meta-Object Compiler,也就是“元对象编译器”,它主要用于处理C源文件中的非标准C代码。Qt 程序在交由标准编…...

关于单体架构缓存刷新实现方案
背景 如果各位看官是分布式项目应该都采用分布式缓存了,例如redis等,分布式缓存不在本次讨论范围哈;我个人建议是,如果是用户量比较大,建议采用分布式缓存机制,后期可以很容易前后到分布式服务或微服务。 …...

洞悉安全现状,建设网络安全防护新体系
一、“网络攻防演练行动“介绍 国家在2016年发布《网络安全法》,出台网络安全攻防演练相关规定:关键信息基础设施的运营者应“制定网络安全事件应急预案,并定期进行演练”。同年“实战化网络攻防演练行动”成为惯例。由公安部牵头࿰…...

spring中怎么通过静态工厂和动态工厂获取对象以及怎么通过 FactoryBean 获取对象
😀前言 本章是spring基于XML 配置bean系类中第4篇讲解spring中怎么通过静态工厂和动态工厂获取对象以及怎么通过 FactoryBean 获取对象 🏠个人主页:尘觉主页 🧑个人简介:大家好,我是尘觉,希望…...

三元组表实现矩阵相加(数据结构)
代码: 含注释,供参考 #include <stdio.h> #include <stdlib.h>typedef struct {int row,col,value;//分别为行数,列数,数值 } Triple; typedef struct {int len;//非零数值的个数Triple data[200]; } TSMatrix;void…...

ChinaJoy 2023微星雷鸟17游戏本震撼发布:搭载AMD锐龙9 7945HX首发8499元
ChinaJoy 2023展会中微星笔记本再次给大家带来惊喜,发布了搭载AMD移动端16大核的旗舰游戏本:雷鸟17,更重要的这样一款旗舰性能的游戏本,首发价8499元堪称当今游戏本市场中的“性价比爆款”! 本着和玩家一同制霸游戏战场…...

各种运算符
算术运算符 1.双目运算符 */%:从左到右优先级依次降低 一些注意事项: 1若a/b都为整型那么结果也为整型,如果ab其中有一个为实型,结果则为实型 求余运算符注意事项: 1运算对象必须为整数 2运算结果的整数跟左边数字的…...

yolov3-tiny原理解析及代码分析
前言 从去年十一月份开始学习yolo神经网络用于目标识别的硬件实现,到现在已经六个月了。一个硬件工程师,C/C基础都差劲的很,对照着darknet作者的源码和网上东拼西凑的原理讲解,一点一点地摸索。刚开始进度很慢,每天都…...

深入了解Redis-实战篇-短信登录
深入了解Redis-实战篇-短信登录 一、故事背景二、知识点主要构成2.1、短信登录2.1.1、生成随机短信验证码引入maven依赖生成验证码 2.1.2、实现登录校验拦截器2.1.3、基于Redis实现短信登录2.1.3.1、发送验证码时存入Redis2.1.3.2、登录时校验验证码 2.1.4、解决状态登录刷新的…...

Mysql的锁
加锁的目的 对数据加锁是为了解决事务的隔离性问题,让事务之前相互不影响,每个事务进行操作的时候都必须先加上一把锁,防止其他事务同时操作数据。 事务的属性 (ACID) 原子性 一致性 隔离性 持久性 事务的隔离级别 锁…...

【EI/SCOPUS征稿】2023年算法、图像处理与机器视觉国际学术会议(AIPMV2023)
2023年算法、图像处理与机器视觉国际学术会议(AIPMV2023) 2023 International Conference on Algorithm, Image Processing and Machine Vision(AIPMV2023) 2023年算法、图像处理与机器视觉国际学术会议(AIPMV2023&am…...

Go语言性能优化建议与pprof性能调优详解——结合博客项目实战
文章目录 性能优化建议Benchmark的使用slice优化预分配内存大内存未释放 map优化字符串处理优化结构体优化atomic包小结 pprof性能调优采集性能数据服务型应用go tool pprof命令项目调优分析修改main.go安装go-wrk命令行交互界面图形化火焰图 性能优化建议 简介: …...

K阶斐波那契数列(数据结构)
代码: 注意k阶斐波那契序列定义:第k和k1项为1,前k - 1项为0,从k项之后每一项都是前k项的和 例如:k2时,斐波那契序列为:0,1,1,2,3,5,8,13... k3时,斐波那契序列为:0,0,…...

【JavaEE】博客系统前后端交互
目录 一、准备工作 二、数据库的表设计 三、封装JDBC数据库操作 1、创建数据表对应的实体类 2、封装增删改查操作 四、前后端交互逻辑的实现 1、博客列表页 1.1、展示博客列表 1.2、博客详情页 1.3、登录页面 1.4、强制要求用户登录,检查用户的登录状态 …...

Redis 简介
文章目录 Redis 简介 Redis 简介 Redis(Remote Dictionary Server),远程词典服务器,基于 C/S 架构,是一个基于内存的键值型 NoSQL 数据库,开源,遵守 BSD 协议,Redis 由 C语言 实现。…...

CS162 13-17 虚拟内存
起源 为啥我们需要虚拟内存-----------需求是啥? 可以给程序提供一个统一的视图,比如多个程序运行同一个代码段的话,同一个kernel,就可以直接共享 cpu眼里的虚拟内存 无限内存的假象 设计迭代过程 为啥这样设计? 一…...

接口自动化测试-Jmeter+ant+jenkins实战持续集成(详细)
目录:导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结(尾部小惊喜) 前言 1、下载安装配置J…...

最长连续序列——力扣128
文章目录 题目描述法一 哈希表 题目描述 法一 哈希表 用一个哈希表存储数组中的数,这样查看一个数是否存在即能优化至 O(1) 的时间复杂度 每次在哈希表中检查是否存在 x−1 即能判断是否需要跳过 int longestConsecutive(vector<int>& nums){unordered_s…...

uniapp app端 echarts 设置tooltip的formatter不生效问题以及解决办法
需求一: y轴数据处理不同数据增加不同单位 需求二: 自定义图表悬浮显示的内容 需求一:实现方式 在yAxis里面添加formatter yAxis: [{//y轴显示value的设置axisLabel: {show: true,formatter (value, index) > {var valueif (value > 1…...

Spring入门-技术简介、IOC技术、Bean、DI
前言 Spring是一个开源的项目,并不是单单的一个技术,发展至今已形成一种开发生态圈。也就是说我们可以完全使用Spring技术完成整个项目的构建、设计与开发。Spring是一个基于IOC和AOP的架构多层j2ee系统的架构。 SpringFramework:Spring框架…...

深度学习之反向传播
0 特别说明 0.1 学习视频源于:b站:刘二大人《PyTorch深度学习实践》 0.2 本章内容为自主学习总结内容,若有错误欢迎指正! 1 forward(前馈运算)过程 通过输入相应的x和权重w(可能涉及bais偏置…...