Node.js是基于事件驱动编程。异步函数随处可见,其中不乏一些常用库的方法。本例就以js中最常见的setTimeout的为例,试图改善一下回调的书写。

先来看一段伪代码:

我们实现一个需求,每隔一段时间打印一段字符串,按照一般的同步设计思维,那么大概是这个样子:

; i < ; i++)
{
    sleep(i);
    printf("out:%d", i);
}

但是在js中可就没这么简单了。首先,类似sleep的setTimeout函数睡眠之后的事件就在回调当中

setTimeout(function(){

},1000)

看到这我想大家也都明白了,这要是写在循环里,很难受,写起来很难受。用递归是一种方法,但是也难受。

解决这个问题的办法要借助node的两样工具:

1.Promise

2.async/await (需要node8 版本)

Promise主要用来优化回调函数的书写格式,使得多次回调的嵌套不会写那么多层。

上述循环的单次循环可以封装成如此格式:

function call_once(_i)
{
    setTimeout(function(_i){
        return ("out:"+_i)
    },_i*1000)
}

Promise不在node官方库中 ,需要使用npm安装,Promise有很多三方包,个人比较常用的是 bluebird

npm install bluebird
const promise = require('bluebird')

对上边的call_once方法封装,就是:

function call_once(_i)
{
    return new promise(function(resolve){
        setTimeout(function(){

            resolve("out:" + _i)

        }, _i*1000)
    })
}

但是这个方法也不能直接当同步方法用,回调后的内容需要写在 promise对象的then方法当中:

call_once(1).then(function(_result){

    console.log(_result);

})

没错,所有在这条主线上执行的内容都需要不断的用这个promise的对象then下去,写在外边是不行的。会提前执行。

所以还是没法把这个东西嵌入到for循环当中,这个时候就需要await/async了。

1.首先把所有的异步内容全部嵌套到一个异步函数当中,声明异步函数需要在function前声明关键字async。

2.在需要阻塞的函数前(这个函数需要返回promise对象)加上await关键字:

async function looper(){
    for(let i = 0; i < 5; i++)
    {
        await call_once(i).then(function(_result)
        {
                console.log(_result)
        })
    }
}

如此就实现了for-sleep的方法。但是要注意,还是只限在lopper()方法当中才是同步的,写在looper外边的内容还是不行。TAT。如果是一些小功能脚本的话,就把looper当main方法用吧。

最后祝您,身体健康。再见

------------------------------------------------------>

to be continued

【Node100In1】01.去异步,解决掉Node.js万恶的回调陷阱的更多相关文章

  1. node.js中的回调

    同步和阻塞:这两个术语可以互换使用,指的是代码的执行会在函数返回之前停止.如果某个操作阻塞,那么脚本就无法继续,这意味着必须等待. 异步和非阻塞:这两个术语可以互换使用,指的是基于回调的.允许脚本并行 ...

  2. node.js如何使用回调

    Node.js到处使用回调,尤其在有I/O(输入/输出)操作的地方. 下面是在一个Node.js中使用filesystem模块中从磁盘上读入文件内容示例一: var fs = require('fs' ...

  3. Node.js标准的回调函数

    Node.js标准的回调函数:第一个参数代表错误信息,第二个参数代表结果. function (err, data) 当正常读取时,err参数为null,data参数为读取到的String.当读取发生 ...

  4. 01慕课网《进击Node.js基础(一)》Node.js安装,创建例子

    版本:偶数位为稳定版本,基数为非稳定版本 - 0.6.x - 0.7.x    - 0.8.x -0.9.x    -0.10.x  -0.11.x 概念:Node.js采用谷歌浏览器的V8引擎,用C ...

  5. Node.js链式回调

    由于异步的关系,代码的书写顺序可能和执行顺序并不一样,可能想先执行A再执行B,但由于异步可能B要先于A执行.例如在OC中使用AFnetworking请求数据然后刷新页面,由于网络请求是用block实现 ...

  6. 【解决】Node JS Error: ENOENT

    The Node Beginner Book 书中的实例代码当上传图片时会报Error: ENOENT, 原因:图片默认会选择系统的缓存文件夹下,在windows下无权访问C盘,所以就报错了.. 解决 ...

  7. node.js异步控制流程 回调,事件,promise和async/await

    写这个问题是因为最近看到一些初学者用回调用的不亦乐乎,最后代码左调来又调去很不直观. 首先上结论:推荐使用async/await或者co/yield,其次是promise,再次是事件,回调不要使用. ...

  8. Node.js 异步异闻录

    本文首发在个人博客:http://muyunyun.cn/posts/7b9fdc87/ 提到 Node.js, 我们脑海就会浮现异步.非阻塞.单线程等关键词,进一步我们还会想到 buffer.模块机 ...

  9. Node.js之异步编程

    > 文章原创于公众号:程序猿周先森.本平台不定时更新,喜欢我的文章,欢迎关注我的微信公众号. ![file](https://img2018.cnblogs.com/blog/830272/20 ...

随机推荐

  1. Mybatis SqlsessionFactory

    在Mybatis 与 Spring 进行整合的时候,我们会进行sqlSessionFactory 的配置,来创建sqlSessionFactory 对象:如下: <bean id="s ...

  2. ieee文献免费下载办法

    sci-hub是个神奇的存在,但突然有短时间不能用了,搜索很久,找到了:https://www.zhihu.com/question/68333471/answer/276287163 这个网址会实时 ...

  3. js二分算法排序

    var arr = [1,2,3,5,10,15,23,35,67,76,78,89,100];var len = arr.length;for (var i = 1; i < len; i++ ...

  4. 1,fiddler的工作原理和安装

    1,工作原理就是通过设置代理监控客户端和服务端的协议 2,fiddler的安装 1,官方的下载地址:https://www.telerik.com/download/fiddler 一步步安装即可 2 ...

  5. Task与线程池

    尽量使用Task,而不是线程池 因为Task是基于线程的,单不是一一对应的 Task的切换与开销要比线程小很多,也更容易管理 http://www.cnblogs.com/yunfeifei/p/41 ...

  6. java Quartz任务调度器

    1.quarz对java1.5实现的简单调度做了封装 /**     * quartz对任务调度进了高度抽象: 1调度器:2任务:3触发器     * Job接口(任务):定义需要调度的任务     ...

  7. IAR FOR AVR 仿真过程中出现全局变量值不断随意变化的问题

    本文记录使用IAR FOR AVR 使用过程中出现的问题,确保自己以后能够有史可查,也分享给遇到同样问题的朋友. 版本信息:IAR Assembler for AVR  5.40.0 (5.40.0. ...

  8. ELK日志系统使用说明

    数据探索 Elasticsearch具有强大的数据检索和分析同能,支持模糊.全文.过滤.管道等数据查询.对于日志型数据处理很有优势. 下图为KIbana的主页图,将逐步说明每一部分的功能: 依照图中的 ...

  9. Kmeanns图片压缩

    from sklearn.datasets import load_sample_image#先导入数据包 china = load_sample_image("china.jpg" ...

  10. 字符串、数组、对象常用API

    常用的字符串API  1.常见方法和属性 length 属性,获取字符串的字符数量 charAt(i) 返回给定位置的字符 charCodeAt( ) 返回给定位置的字符的字符编码 <scrip ...