如何优雅地退出函数

退出函数怎么写?有人会说一个 return 就退出函数了,有这么简单吗?来看一个表单校验和提交结果的函数:

js 复制代码
// 表单校验
function validate() {}
// 提交表单
function submit() {
  validate();// ?如果表单校验失败怎么退出函数呢?
  fetch("/api/submit", {
    // ...
  });
}

如果表单校验失败怎么退出函数呢?传统的思维方式必然是使用return解决,让 validate 函数返回一个值来判断是退出函数还是继续执行:

diff 复制代码
function validate() {}
function submit() {
+  if (!validate()) {
+    return;
+  }
  fetch("/api/submit", {
    // ...
  });
}

这样解决是可以的,但是不够优雅,为什么呢?因为 validate 函数本身是不需要返回值的,现在却为了功能必须添加一个返回值,并且 submit 函数还依赖于 validate 的返回值,增加了函数之间的耦合度。如何优雅地退出函数,这就是今天的主题。

抛出异常

第一个方法当然是抛出异常,这一招我屡试不爽,还是上面那个案例,先看看如何通过抛出异常退出函数:

diff 复制代码
// 表单校验
function validate() {
+    throw new Error("验证失败");
}
// 提交表单
function submit() {
  validate();// ?如果表单校验失败怎么退出函数呢?
  fetch("/api/submit", {
    // ...
  });
}

如果 submit 还有其他很多流程并且这些流程中的某个分支也需要退出整个 submit 函数,使用抛出异常退出函数的话就不需要写那么多的 if else 了,例如:

js 复制代码
// 提交表单
function submit() {
  validate();// ?如果表单校验失败怎么退出函数呢?
  doSomething1(); 
  doSomething2();
  fetch("/api/submit", {
    // ...
  });
}

但是这种方式也有一个弊端,那就是带来了控制台的爆红,不过一看就看得出来这不是程序报错导致的,是我们手动抛出的:

抛出异常这种方式在异步函数中同样适用,但是需要改变写法,先来看看不改变写法会怎么样:

diff 复制代码
+ async function validate() {
  throw new Error("validate failed");
}
+ async function submit() {
  validate();
  fetch("/api/submit", {
    // ...
  });
}

咦,为什么抛出异常不生效了呢,要解答这个问题,首先得把这段代码"翻译"一下:

diff 复制代码
function validate() {
+  return Promise.reject(new Error("validate failed"));
}

显然这里函数没有真正抛出异常,而是被 Promise 拦截掉了,导致退出函数失败。这里只需要增加一个 await 就可以退出函数了:

diff 复制代码
async function submit() {
+  await validate();
  fetch("/api/submit", {
    // ...
  });
}

异步函数 reject

上面说了抛出异常可以退出异步函数,但是不仅仅这一种方式,还可以通过 Promise.reject 退出异步函数:

diff 复制代码
async function validate() {
  return Promise.reject();
}

所以如果想退出函数可以先将函数改为 async 函数,然后返回一个 reject 状态的 Promise,这样就能够退出函数了。

最后,总结一下所有退出函数的方法:

  1. 函数返回一个布尔值,根据这个布尔值判断是否 return 退出函数
  2. 抛出异常退出函数,注意异步函数需要加上 await,否则无法退出
  3. 将普通函数转为异步函数,然后返回一个 reject 状态的 Promise,也可以实现退出函数的目的
相关推荐
zhougl9962 小时前
html处理Base文件流
linux·前端·html
花花鱼2 小时前
node-modules-inspector 可视化node_modules
前端·javascript·vue.js
HBR666_2 小时前
marked库(高效将 Markdown 转换为 HTML 的利器)
前端·markdown
careybobo3 小时前
海康摄像头通过Web插件进行预览播放和控制
前端
TDengine (老段)4 小时前
TDengine 中的关联查询
大数据·javascript·网络·物联网·时序数据库·tdengine·iotdb
杉之5 小时前
常见前端GET请求以及对应的Spring后端接收接口写法
java·前端·后端·spring·vue
喝拿铁写前端5 小时前
字段聚类,到底有什么用?——从系统混乱到结构认知的第一步
前端
再学一点就睡5 小时前
大文件上传之切片上传以及开发全流程之前端篇
前端·javascript
木木黄木木6 小时前
html5炫酷图片悬停效果实现详解
前端·html·html5
请来次降维打击!!!7 小时前
优选算法系列(5.位运算)
java·前端·c++·算法