前言:文章由本人在学习之余总结巩固思路,不足之前还请指出。

一.异步编程

首先我们先简单来回顾一下同步API和异步API的概念

1.同步API:只有当前的API执行完成之前,才会执行下一个API

例:

console.log(‘first');
console.log('last);
结果:
first
last

2.异步API:当前API的执行不会阻塞后续代码的执行

例:

console.log('first');
setTimeout(
() => { console.log('last');
}, 2000);
console.log('middle');
结果:
first
middle
last

执行顺序分析:

首先脚本会先执行同步代码,这时有一个同步代码区,按着从上到下的顺序进行。当所有的同步代码执行完成之后,再进入异步代码区查找是否有异步代码,并开始执行,执行完之后,调用对应的回调函数。

拿例子中的计时器来说,该异步函数每2秒都会重新调用一次。

注意:即使是计时器的时间设置为0,它任是一个异步API,不会随着同步一起执行。

3.Node.js中的异步API

fs.readFile('./demo.txt', (err, result) => {});

当硬盘读取了文件之后,调用后方的回调函数;

但这个时候,我们有了一个需求,依次读取A,B,C三个文件。由于文件大小不一定是我们知道的,所以读取,查找的速度我们也并不知道。

按照同步的思路来写的话:

fs.readFile('./1.txt', 'utf8', (err, result1) => {console.log(result1)});
fs.readFile('./2.txt', 'utf8', (err, result2) => {console.log(result2)});
fs.readFile('./3.txt', 'utf8', (err, result3) => {console.log(result3)});

结果未必如我们所愿。

这时我们也许会想到一个办法,既然该API是异步API,我们把各个API嵌套起来,这样不久可以依次执行了?

fs.readFile('./1.txt', 'utf8', (err, result1) => {
console.log(result1)
fs.readFile('./2.txt', 'utf8', (err, result2) => {
console.log(result2)
fs.readFile('./3.txt', 'utf8', (err, result3) => {
console.log(result3)
})
})
});

emmm....确实可以,但他的缺点也能预料到:这里只有三个文件,可能不易看出其劣势,但是如果是300个呢?代码的可读性将大幅度降低,维护的难度相应提高,这是我们不愿意看到的。

这一现象,我们称其为回调地狱(callbackhell)。进来了就si里面了。

解决的办法当然也有,这时该轮到我们的Promise登场了。

二.Promise

Promise出现的目的是解决Node.js异步编程中回调地狱的问,它是一个构造函数,我们要用new Promise的方法调用。

我们先来简单地介绍一下Promise。

let promise = new Promise((resolve, reject) => {});
Promise的参数为一个匿名回调函数,其中reslove,reject也是回调函数,他能将异步API的执行和结果进行分离,reslove对应着result(正常思路下)当fs有返回结果的时候,我们可以将其通过回调函数的方式将其发送到外面,
同理,当fs出现错误的时候,我们可以将其发送到外面进行处理。这里要用到Promise下面的两个方法promise.then()&promise.catch(),分别用来对结果和错误信息进行处理。 我们结合实例来分析这些回调函数。

let promise = new Promise((resolve, reject) => {

    fs.readFile('./1.txt', 'utf8', (err, result) => {

        if (err != null) {
reject(err);
  相当于执行then里面的回调函数
}else {
resolve(result);
  相当于执行catch里面的回调函数
}
});
});

  promise.then((result) => {
    console.log(result);
  })
  .catch((err)=> {
    console.log(err);
  })


用此方法来对我们之前的函数进行包装,分析一下,既然我们有三个异步API,我们则需要用三个Promise将他们包裹起来,我们需要让这三个promise依次执行,但是如果我们直接声明了一个变量等于promise的话就直接执行了,所以
这里我们用一个函数把他封装起来
function p1 () {
return new Promise ((resolve, reject) => {
fs.readFile('./1.txt', 'utf8', (err, result) => {
resolve(result)
})
});
} function p2 () {
return new Promise ((resolve, reject) => {
fs.readFile('./2.txt', 'utf8', (err, result) => {
resolve(result)
})
});
} function p3 () {
return new Promise ((resolve, reject) => {
fs.readFile('./3.txt', 'utf8', (err, result) => {
resolve(result)
})
});
}

为了实现顺序调用,我们使用链式编程

p1().then((r1)=> {
console.log(r1);
return p2();
})
.then((r2)=> {
console.log(r2);
return p3();
})
.then((r3) => {
console.log(r3)
})

注意:这里return 调用返回的结果我们可以参考MDN的解释

返回一个已经是接受状态的 Promise,那么 then 返回的 Promise 也会成为接受状态,并且将那个 Promise 的接受状态的回调函数的参数值作为该被返回的Promise的接受状态回调函数的参数值。

这里看上去我们的代码的似乎比之前的嵌套关系更为复杂,这里就要引入我们的异步函数了。

3.异步函数

异步函数是异步编程语法的终极解决方案,它可以让我们将异步代码写成同步的形式,让代码不再有回调函数嵌套,使代码变得清晰明了。

举一个简单的函数来看看他的用法

// 1.在普通函数定义的前面加上async关键字 普通函数就变成了异步函数
// 2.异步函数默认的返回值是promise对象
// 3.在异步函数内部使用throw关键字进行错误的抛出 async function fn () {
这里我们发现我们不需要再new一个Promise再将其返回了
return 123; //正常的时候用return
// throw '发生了一些错误';出错的时候throw
} console.log(fn ())
fn ()
.then(function (data) {
console.log(data);
})
.catch(function (err){
console.log(err);
})
那么如何让我们的函数有序地进行呢?这里我们不用再采用return嵌套,接下来就要用到await了
我们定义一个run函数
// await关键字
// 1.它只能出现在异步函数中
// 2.await promise 它可以暂停异步函数的执行 等待promise对象返回结果后再向下执行函数
async function run () {
let r1 = await p1()
let r2 = await p2()
// await不能直接得到throw并赋值给r3这里我们采用catch试试
let r3 = p3().catch(n => console.log(n));// p3
console.log(r1)
console.log(r2)
console.log(r3 instanceof Promise)//ture
// console.log(r3)
}
这样就能实现我们的按顺序执行了
此处r3的值是我在记笔记的时候发现await并不直接接受reject的Promise,所以做了个输出的尝试,随意看看就好
run();
新手一枚,大家有啥好的想法或者问题欢迎一起讨论
												

简述异步编程&Promise&异步函数的更多相关文章

  1. 简单实现异步编程promise模式

    本篇文章主要介绍了异步编程promise模式的简单实现,并对每一步进行了分析,需要的朋友可以参考下 异步编程 javascript异步编程, web2.0时代比较热门的编程方式,我们平时码的时候也或多 ...

  2. 前端分享----JS异步编程+ES6箭头函数

    前端分享----JS异步编程+ES6箭头函数 ##概述Javascript语言的执行环境是"单线程"(single thread).所谓"单线程",就是指一次只 ...

  3. 异步编程——promise

    异步编程--promise 定义 Promise是异步编程的一个解决方案,相比传统的解决方法--回调函数,使用Promise更为合理和强大,避免了回调函数之间的层层嵌套,也使得代码结构更为清晰,便于维 ...

  4. 异步编程promise

    异步编程发展 异步编程经历了 callback.promise.async/await.generator四个阶段,其中promise和async/await使用最为频繁,而generator因为语法 ...

  5. C#复习笔记(5)--C#5:简化的异步编程(异步编程的基础知识)

    异步编程的基础知识 C#5推出的async和await关键字使异步编程从表面上来说变得简单了许多,我们只需要了解不多的知识就可以编写出有效的异步代码. 在介绍async和await之前,先介绍一些基础 ...

  6. .NET “底层”异步编程模式——异步编程模型(Asynchronous Programming Model,APM)

    本文内容 异步编程类型 异步编程模型(APM) 参考资料 首先澄清,异步编程模式(Asynchronous Programming Patterns)与异步编程模型(Asynchronous Prog ...

  7. 异步编程Promise/Deferred、多线程WebWorker

    长期以来JS都是以单线程的模式运行的,而JS又通常应用在操作用户界面和网络请求这些任务上.操作用户界面时不能进行耗时较长的操作否则会导致界面卡死,而网络请求和动画等就是耗时较长的操作.所以在JS中经常 ...

  8. Javascript异步编程之一异步原理

    本系列的例子主要针对node.js环境,但浏览器端的原理应该也是类似的. 本人也是Javascript新手,把自己这段时间学习积累的要点总结下来,希望可以对同样在学习Javascript/node.j ...

  9. C#复习笔记(5)--C#5:简化的异步编程(异步编程的深入分析)

    首先,阐明一下标题的这个“深入分析”起得很惭愧,但是又不知道该起什么名字,这个系列也主要是做一些复习的笔记,供自己以后查阅,如果能够帮助到别人,那自然是再好不过了. 然后,我想说的是异步方法的状态机真 ...

随机推荐

  1. ATcoder--D - Summer Vacation

    这个题目的题意有点难搞 题目连接: https://atcoder.jp/contests/abc137/tasks/abc137_d 题目大意:输入n和m 指的是一共有n个输入在m天前一共能赚到的钱 ...

  2. pomelo环境配置(windows环境)

    目录 简介 准备 安装 工程的创建 简介 1.网易开源,免费,业(diao)界(si)良(fu)心(li)呀,^.^ 2.游戏服务器框架(当然也可以用于web服务器) 3.高性能.高可伸缩.分布式,多 ...

  3. requets中urlencode的问题

    前言 今天团队群里有师傅问requests怎么设置不解码,这里是语误,其实师傅想说的是,如果设置不编码. 一开始我没懂,然后师傅们解答了这个问题后,我想了会儿懂了. 在一些CTF题目中,可能会碰到这样 ...

  4. keras API的使用,神经网络层,优化器,损失函数,查看模型层数,compile和fit训练

    layers介绍 Flatten和Dense介绍 优化器 损失函数 compile用法 第二个是onehot编码 模型训练 model.fit  两种创建模型的方法 from tensorflow.p ...

  5. 【认证与授权】2、基于session的认证方式

    这一篇将通过一个简单的web项目实现基于Session的认证授权方式,也是以往传统项目的做法. 先来复习一下流程 用户认证通过以后,在服务端生成用户相关的数据保存在当前会话(Session)中,发给客 ...

  6. 三本毕业(非科班),四次阿里巴巴面试,终拿 offer(大厂面经)

    作者:gauseen 原文:https://github.com/gauseen/blog 公众号:「学前端」,只搞技术不搞广告文,欢迎关注~ 第一次 20:00 电话一面 - 自我介绍 - 对公司工 ...

  7. 从零开始装CentOS以及配置Redis,前端都可以!!!

    ##### 从零开始装CentOS以及配置Redis 1.新建虚拟机 --- ![image](https://img2018.cnblogs.com/blog/1334966/201910/1334 ...

  8. 编程是要偷懒的--option简练写法

    没改前: if(!empty($search)){ $where['personal_name'] = array('like','%'. $search . '%'); $this -> as ...

  9. PHP 使用try catch,捕获异常

    <?php     header('Content-type:text/html;charset=utf-8');    $a = 1;    $b = 2;    try {        / ...

  10. java学习(第四篇)数组

    一.一维数组 1.声明,分配内存 int[] a=new int[10]; 数组元素的数据类型 [] 数组名=new 类型 [数组元素个数]: 2.初始化 int[] a=new int[] {1,2 ...