NodeJS实现同步的方法


Posted in NodeJs onMarch 02, 2019

NodeJS被打上了单线程、非阻塞、事件驱动…..等标签。

在单线程的情况下,是无法开启子线程的。经过了很久的研究,发现并没有thread函数!!!但是有时候,我们确实需要“多线程”处理事务。nodeJS有两个很基础的api:setTimeout和setInterval。这两个函数都能实现“异步”。

nodeJS的异步实现:nodeJS有一个任务队列,在使用setInterval函数的时候,会每隔特定的时间向该任务队列增加任务,从而实现“多任务”处理。但是,“特定的时间”不代表是具体的时间,也有可能是会大于我们设定的时间,也有可能小于。

我们跑跑下面代码块

setInterval(function() {
  console.log(new Date().getTime());
}, 1000);

输出的结果如下:

1490531390640
1490531391654
1490531392660
1490531393665
1490531394670
1490531395670
1490531396672
1490531397675
......

我们可以看到,所有的时间间隔都是不一样的。时间的偏移不仅包含了间隔的1s,还包含了console.log()的耗时,以及new Date()的耗时。在大量的数据统计下,时间间隔近似于1s。

问题来了,setInterval是能实现多任务的效果,但是怎样才能实现任务之间的同步操作呢?这里实现的方法是通过回调函数实现的。

function a(callback) {
  // 模拟任务a耗时
  setTimeout(function() {
    console.log("task a end!");
    // 回调任务b
    callback();
  }, 3000);
};
function b() {
  setTimeout(function() {
    console.log("task b end!");
  }, 5000);
}
a(b);

这里举了一个很简单的例子,就是将b方法的实现赋值给a方法的callback函数从而实现函数回调,但是会有个问题。假设a方法依赖于b方法,b方法依赖于c方法,c方法依赖于d方法…..也就意味着每个方法的实现都需要持有上一个方法的实例,从而实现回调。

function a(b, c, d) {
  console.log("hello a");
  b(c, d);
};
function b(c, d) {
  console.log("hello b");
  c(d);
};
function c(d) {
  console.log("hello c");
  d()
};
function d() {
  console.log("hello d");
};
a(b, c, d);

输出结果

hello a
hello b
hello c
hello d

如果回调函数写的多了,会造成代码特别特别恶心。

如果有类似于sync的函数能让任务顺序执行就更好了。终于找到了async这个库

$ npm instanll async

async = require("async");
a = function (callback) {
  // 延迟5s模拟耗时操作
  setTimeout(function () {
    console.log("hello world a");
    // 回调给下一个函数
    callback(null, "function a");
  }, 5000);
};
b = function (callback) {
  // 延迟1s模拟耗时操作
  setTimeout(function () {
    console.log("hello world b");
    // 回调给下一个函数
    callback(null, "function b");
  }, 1000);
};
c = function (callback) {
  console.log("hello world c");
  // 回调给下一个函数
  callback(null, "function c");
};
// 根据b, a, c这样的顺序执行
async.series([b, a, c], function (error, result) {
  console.log(result);
});

注释基本能够很好的理解了,我们看看输出

hello world b
hello world a
hello world c
[ 'function b', 'function a', 'function c' ]

上面的基本async模块的实现的如果了解更多关于async模块的使用,可以查询官方文档

其实nodeJS基本api也提供了异步实现同步的方式。基于Promise+then的实现

sleep = function (time) {
  return new Promise(function () {
    setTimeout(function () {
      console.log("end!");
    }, time);
  });
};
console.log(sleep(3000));

输出结果为:

Promise { <pending> }
end!

可以看出来,这里返回了Promise对象,直接输出Promise对象的时候,会输出该对象的状态,只有三种:PENDING、FULFILLED、REJECTED。字面意思很好理解。也就是说Promise有可能能实现我们异步任务同步执行的功能。我们先用Promise+then结合起来实现异步任务同步操作。

sleep = function () {
  return new Promise(function (resolve, reject) {
    setTimeout(function () {
      console.log("start!");
      resolve();
    }, 1000);
  })
    .then(function () {
      setTimeout(function () {
        console.log("end!");
      }, 2000);
    })
    .then(function () {
      console.log("end!!");
    })
};
console.log(sleep(1000));

输出结果:

Promise { <pending> }
start!
end!!
end!

在new Promise任务执行完后,调用了resolve才会执行所有的then函数,并且这些then函数是异步执行的。由输出结果可以知道。(如果所有then是顺序执行的应该是end! -> end!!)。但是上述也做到了两个异步任务之间顺序执行了。

不过,还有更加优雅的方式:使用async+await。

display = function(time, string) {
  return new Promise(function (resovle, reject) {
    setTimeout(function () {
      console.log(string);
      resovle();
    }, time)
  });
};
// 执行顺序:b a c
fn = async function () {
  // 会造成阻塞
  await display(5000, "b");
  await display(3000, "a");
  await display(5000, "c");
}();

输出结果:

b
a
c

由于这里时间输出比较尴尬,只能通过我们来感知,本人通过个人“感知”知道了在display b过度到display a的时候大概用了3s,再过度到display c的时候大概用了5s

这里需要声明的是:才开始正式的学习nodeJS。在学习的过程中,做的笔记,很多概念与理解可能会错误,如果错误,欢迎纠正。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对三水点靠木的支持。如果你想了解更多相关内容请查看下面相关链接

NodeJs 相关文章推荐
nodejs中exports与module.exports的区别详细介绍
Jan 14 NodeJs
nodejs 实现模拟form表单上传文件
Jul 14 NodeJs
nodeJS代码实现计算交社保是否合适
Mar 09 NodeJs
Ajax异步文件上传与NodeJS express服务端处理
Apr 01 NodeJs
Nodejs进阶之服务端字符编解码和乱码处理
Sep 04 NodeJs
nodejs使用redis作为缓存介质实现的封装缓存类示例
Feb 07 NodeJs
nodejs 使用http进行post或get请求的实例(携带cookie)
Jan 03 NodeJs
Nodejs对postgresql基本操作的封装方法
Feb 20 NodeJs
Nodejs中的require函数的具体使用方法
Apr 02 NodeJs
NodeJS读取分析Nginx错误日志的方法
May 14 NodeJs
nodejs处理tcp连接的核心流程
Feb 26 NodeJs
PHPStorm中如何对nodejs项目进行单元测试详解
Feb 28 #NodeJs
Nodejs对postgresql基本操作的封装方法
Feb 20 #NodeJs
深入理解nodejs搭建静态服务器(实现命令行)
Feb 05 #NodeJs
Nodejs实现的操作MongoDB数据库功能完整示例
Feb 02 #NodeJs
基于Koa(nodejs框架)对json文件进行增删改查的示例代码
Feb 02 #NodeJs
用Electron写个带界面的nodejs爬虫的实现方法
Jan 29 #NodeJs
NVM安装nodejs的方法实用步骤
Jan 16 #NodeJs
You might like
电脑硬件及电脑配置知识大全
2020/03/17 数码科技
分享php多功能图片处理类
2016/05/15 PHP
javascript实现上传图片前的预览(TX的面试题)
2007/08/20 Javascript
JS支持带x身份证号码验证函数
2008/08/10 Javascript
Javascript中的变量使用说明
2010/05/18 Javascript
用js判断页面刷新或关闭的方法(onbeforeunload与onunload事件)
2012/06/22 Javascript
jQuery中:checked选择器用法实例
2015/01/04 Javascript
jQuery easyui的validatebox校验规则扩展及easyui校验框validatebox用法
2016/01/18 Javascript
基于jQuery实现表格内容的筛选功能
2016/08/21 Javascript
JSONP跨域请求
2017/03/02 Javascript
JS中正则表达式全局匹配模式 /g用法详解
2017/04/01 Javascript
使用nodejs+express实现简单的文件上传功能
2017/12/27 NodeJs
详解在React里使用&quot;Vuex&quot;
2018/04/02 Javascript
vue组件实现可搜索下拉框扩展
2020/10/23 Javascript
js实现购物车功能
2018/06/12 Javascript
详解webpack+ES6+Sass搭建多页面应用
2018/11/05 Javascript
全面了解JavaScript的作用域链
2019/04/03 Javascript
vue路由切换时取消之前的所有请求操作
2020/09/01 Javascript
[00:06]Yes,it worked!小卡尔成功穿越时空加入战场!
2019/07/20 DOTA
[01:05:59]Mineski vs Secret 2019国际邀请赛淘汰赛 败者组 BO3 第二场 8.22
2019/09/05 DOTA
[01:11:46]DOTA2-DPC中国联赛 正赛 iG vs Magma BO3 第一场 2月23日
2021/03/11 DOTA
Python中多线程thread与threading的实现方法
2014/08/18 Python
python的paramiko模块实现远程控制和传输示例
2017/10/13 Python
Android基于TCP和URL协议的网络编程示例【附demo源码下载】
2018/01/23 Python
python正则实现提取电话功能
2018/02/24 Python
Python函数装饰器实现方法详解
2018/12/22 Python
Django网络框架之HelloDjango项目创建教程
2019/06/06 Python
利用pipenv和pyenv管理多个相互独立的Python虚拟开发环境
2020/11/01 Python
意大利团购网站:Groupon意大利
2016/10/11 全球购物
个性化皮包、小袋、生活配件:Mon Purse
2019/03/26 全球购物
自我评价格式
2014/01/06 职场文书
人力资源部副职的竞聘演讲稿
2014/01/07 职场文书
秋季开学典礼主持词
2014/03/19 职场文书
2014年人力资源部工作总结
2014/11/19 职场文书
毕业生个人总结
2015/02/28 职场文书
《蟋蟀的住宅》教学反思
2016/02/17 职场文书