NodeJS研究笔记:异步编程导致难以察觉的bug_tyler_download的博客-程序员宅基地

技术标签: javascript 服务端  web  异步  nodejs  javascript  

NodeJS 一显著特点是它的异步处理特性。例如,当我读入一个文件时,Node会将文件读取操作放在后台处理,当程序提交读取请求后,不必堵塞在原地等待请求完成,而是提供一个请求完成的回调函数,然后接着进行其他工作,一旦文件读取完成后,我们提供的函数会被及时调用,在回调函数中,再对读到的文件内容进行处理。正是由于这种异步处理机制,使得Node成为后台开发的一大利器,特别是其异步处理框架提供的高效率,使得它最近成为web后台开发的重要选择。


但是,灵活性是有成本的,异步处理模式提高了程序设计逻辑的复杂性,一旦稍有粗心大意,异步流程会导致一些难以察觉的bug, 例如下面的例子:

var EventEmitter = require('events').EventEmitter;

function doSomeThingSlow() {
	var events = new EventEmitter();
	 events.emit('success');
	
	return events;
}

doSomeThingSlow().on('success', function() {
	console.log('success!');
});

在doSomeThingSlow  中,通过EventEmitter向消息队列发生了一个事件消息叫success, 在底部,我们通过emitter 的on函数,监听success消息,一旦发现该消息,我们做相应的处理。但这段程序有bug, 问题就是‘success'消息的处理函数不会被执行,也就是console.log('success!'); 无法执行。主要问题在于当doSomeThingSlow执行时,在events.emit将success事件触发时,程序还没有运行到底部的doSomeThingSlow.on(...), 也就是success事件触发时,我们来不及为该事件注册对应的处理函数。但程序执行到底部的on部分时,才开始对'success'事件注册处理函数,但此时,事件已经触发完毕,因此也就错过了事件的处理机会。


解决该问题的方法是,利用 Node的异步处理机制,将events.emit('success') 这一触发动作放入到消息队列中异步执行:

var EventEmitter = require('events').EventEmitter;

function doSomeThingSlow() {
	var events = new EventEmitter();
	process.nextTick(function() {
		events.emit('success');
	});
	 
	
	return events;
}

doSomeThingSlow().on('success', function() {
	console.log('success!');
});

process.nextTick 会将给定的函数添加到loop循环的下一个执行队列中,也就是说,nextTick 会将传给它的函数添加到主循环的待处理队列中等待处理,然后立刻返回,这样on内部的代码会先于events.emit被执行,也就是在事件触发前,事件的处理函数会先被注册,这样当events.emit('success')执行时,console.log('success!')就会被执行。


所以,如果运行上一个程序,我们在控制台看不到打印信息,运行第二个程序,我们才能看到打印信息。此类bug比较诡异,如果对Node的异步处理机制理解不深的话,就很难发现这种问题。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/tyler_download/article/details/51072021

智能推荐

java笔记_笑容温暖城寨的博客-程序员宅基地

安卓课的Java扩展笔记包类命名规范总结输出换行与不换行mian方法中的args参数Math类快捷输出语句static关键字定义无参方法定义带返回值方法包package com.eoe.basic.day01;//包名 在java当中用.表示windos中的/,一般格式:com.公司名.项目名.业务模块名称例如:com.sina.crm.user类命名规范总结1、java中可以有多个类2、java中可以有多个类,但是最多只有一个类的类名和文件名相同3、如果一个类被public修饰,那该类

一、网络游戏架构的前世今生(2)_王元恺David的博客-程序员宅基地

网络游戏架构演进史,优化游戏网络,自定义网络协议

在Chrome 浏览器上滚动截屏_ianly梁炎的博客-程序员宅基地_横向滚动截图

浏览器滚动截屏,截取整个页面 在Chrome 浏览器上滚动截屏,不需要插件和任何 app,利用 chrome 浏览器原生功能即可实现。打开 Chrome 浏览器,进入需要截图的网站页面打开开发者工具:在页面任何地方点击鼠标右键,在弹出菜单中选择「检查」选项。或者使用快捷键组合:option + command + i。打开命令行(command palette):command + s...

微信小程序 获取用户网络状态和设备的信息_东边的小山的博客-程序员宅基地

var app = getApp()Page({ data: { motto: 'Hello World', userInfo: {}, netWorkType: '', phoneType: '', phoneSystemType: '', }, //事件处理函数 bindViewTap: function() { wx.nav...

CentOS7使用firewall-cmd打开关闭防火墙与端口_purple.taro的博客-程序员宅基地_firewall-cmd 关闭防火墙

先查看防火墙是否开启了1521端口:firewall-cmd --permanent --query-port=1521/tcp打印结果如下:no表示没有开放1521端口,那么添加下该端口:firewall-cmd --permanent --add-port=1521/tcp打印结果如下:success重新加载防火墙策略:firewall-cmd --reload执行成功后,查看1521端口是否被开启:firewall-cmd --permanent --query-port=152

MySQL、Oracle 通过SQL查看表注释、字段信息_白衣若尘的博客-程序员宅基地

MySQL: 查看表注释: 用 SHOW TABLE STATUS [FROM db_name] 示例: SHOW TABLE STATUS ; --数据库下所有表注释 SHOW TABLE STATUS FROM d

随便推点

大神博客地址_骏马传奇的博客-程序员宅基地

骏马金龙 http://www.cnblogs.com/f-ck-need-u/ 散尽浮华 http://www.cnblogs.com/kevingrace/ kris12 https://www.cnblogs.com/shengyang17/default.html?page=11 Edison Zhou https://www.cnblogs.com/...

圆排列问题_小张的java日记的博客-程序员宅基地_圆排列java

问题给定n个圆的半径序列,将它们放到矩形框中,各圆与矩形底边相切,求具有最小排列长度的圆排列。解析圆排列问题的解空间是一棵排列树。按照回溯法搜索排列树的算法框架,设开始时a=[r1,r2,……rn]是所给的n个元的半径,则相应的排列树由a[1:n]的所有排列构成。 首先计算圆在当前圆排列中的横坐标,由x^2 = sqrt((r1+r2)^2-(r1-r2)^2)推导出x = 2 * sqrt(r1 * r2)。然后计算当前圆排列的长度。变量lenmin记录当前最小圆排列长度...

获取颜色编码_Java_Hello_World.的博客-程序员宅基地

1,使用截屏功能2,正在截屏的时候按下shit,然后看到一个#c92027(red/红色)3,按下字母C4,到需要颜色代码的地方Ctrl+V

python 函数进阶_weixin_30951231的博客-程序员宅基地

三元运算a = 1b = 5c = a if a>b else bprint(c)构成变量 = 条件返回True的结果 if 条件 else 条件返回False的结果必须要有返回的结果必须要有if和else只能是简单的情况def func(a,b): return a if a>b else bc = func(3,5)...

rqt_graph提示b‘Format: “dot“ not recognized. Use one of:\n‘_qq_46145354的博客-程序员宅基地

在终端运行roscore,分别运行rosrun turtlesim turtle和rosrun turtlesim turtle_teleop_key,再输入rqt_graph,提示如下错误"dot" with args ['-Tdot', '/tmp/tmp145cvdav'] returned code: 1stdout, stderr:b''b'Format: "dot" not recognized. Use one of:\n'PluginHandlerDirect._resto.

form标签_ailihx的博客-程序员宅基地

这个标签会生成HTML form标签,同时为form内部所包含的标签提供一个绑定路径(binding path)。 它把命令对象(command object)存在PageContext中,这样form内部的标签 就可以使用这个对象了。标签库中的其他标签都声明在form标签的内部。 让我们假设有一个叫User的领域对象,它是一个JavaBean,有着诸如 firstName和lastName这