浅谈node.js 回调函数与事件机制

node.js 是单进程单线程应用程序,但通过事件和回调支持并发,所以性能非常高。

node.js 的每一个API都是异步的,并作为一个独立线程运行,使用异步函数调用,并处理并发。

node.js 基本上所有的事件机制都是用设计模式中观察者模式实现。node.js 单线程类似进入一个while(true)的事件循环,直到没有事件观察者退出,每个异步事件都生成一个事件观察者,如果有事件发生就调用该回调函数。

 

1、回调函数

node.js 异步编程的直接体现就是回调。异步编程依托于回调来实现,但不能说使用了回调后程序就异步化了。

回调函数在完成任务后就会被调用,node.js 使用了大量的回调函数,node.js 的所有 API 都支持回调函数。

阻塞代码:

mainSync.js:

var fs = require("fs");
var data = fs.readFileSync('inputSync.txt');
console.log(data.toString());
console.log("程序执行结束!");

运行方式:node mainSync.js

非阻塞代码:

mainAsyn.js:

var fs = require("fs");
fs.readFile('inputAsyn.txt', function (err, data) {
if (err)
return console.error(err);
    	console.log(data.toString());
});
console.log("程序执行结束!");

运行方式:node mainAsyn.js

 

2、事件机制

node.js 使用事件驱动模型,当web server接收到请求,就把它关闭然后进行处理,然后去服务下一个web请求。当这个请求完成,它被放回处理队列,当到达队列开头,这个结果被返回给用户。在事件驱动模型中,会生成一个主循环来监听事件,当检测到事件时触发回调函数。

node.js 有多个内置的事件,可以通过引入 events 模块,并通过实例化 EventEmitter 类来绑定和监听事件。

main.js:

// 引入 events 模块
var events = require('events');
// 创建 eventEmitter 对象
var eventEmitter = new events.EventEmitter();
// 创建事件处理程序
var connectHandler = function connected() {
   	console.log('连接成功。');
   	// 触发 data_received 事件
   	eventEmitter.emit('data_received');
}
// 绑定 connection 事件处理程序
eventEmitter.on('connection', connectHandler);
// 使用匿名函数绑定 data_received 事件
eventEmitter.on('data_received', function(){
   	console.log('数据接收成功。');
});
// 触发 connection 事件
eventEmitter.emit('connection');
console.log("程序执行完毕。");

运行方式:

node main.js

运行结果:

连接成功。

数据接收成功。

程序执行完毕。

 

3、EventEmitter类

events 模块只提供了一个对象: events.EventEmitter。EventEmitter 的核心就是事件触发与事件监听器功能的封装。

EventEmitter 对象如果在实例化时发生错误,会触发 'error' 事件。当添加新的监听器时,'newListener' 事件会触发,当监听器被移除时,'removeListener' 事件被触发。

EventEmitter 的每个事件由一个事件名和若干个参数组成,事件名是一个字符串,通常表达一定的语义。对于每个事件,EventEmitter 支持 若干个事件监听器。当事件触发时,注册到这个事件的事件监听器被依次调用,事件参数作为回调函数参数传递。

main.js:

var events = require('events');
var emitter = new events.EventEmitter();
emitter.on('someEvent', function(arg1, arg2) {
	console.log('listener1', arg1, arg2);
});
emitter.on('someEvent', function(arg1, arg2) {
	console.log('listener2', arg1, arg2);
});
emitter.emit('someEvent', 'arg1 参数', 'arg2 参数');

运行方式:

node main.js

运行结果:

listener1 arg1 参数 arg2 参数

listener2 arg1 参数 arg2 参数

 

1) 方法

序号

方法 & 描述

1

addListener(event, listener)
为指定事件添加一个监听器到监听器数组的尾部。

2

on(event, listener)
为指定事件注册一个监听器,接受一个字符串 event 和一个回调函数。

server.on('connection', function (stream) {

  console.log('someone connected!');

});

3

once(event, listener)
为指定事件注册一个单次监听器,即监听器最多只触发一次,触发后立刻解除该监听器。

server.once('connection', function (stream) {

  console.log('Ah, we have our first user!');

});

4

removeListener(event, listener)
移除指定事件的某个监听器,监听器 必须是该事件已经注册过的监听器。

var callback = function(stream) {

  console.log('someone connected!');

};

server.on('connection', callback);// ...

server.removeListener('connection', callback);

5

removeAllListeners([event])
移除所有事件的所有监听器, 如果指定事件,则移除指定事件的所有监听器。

6

setMaxListeners(n)
默认情况下,EventEmitters 如果你添加的监听器超过10个就会输出警告信息。 setMaxListeners 函数用于提高监听器的默认限制的数量。

7

listeners(event)
返回指定事件的监听器数组。

8

emit(event, [arg1], [arg2], [...])
按参数的顺序执行每个监听器,如果事件有注册监听返回 true,否则返回 false。

 

2) 类方法

序号

方法 & 描述

1

listenerCount(emitter, event)
返回指定事件的监听器数量。

 

3) 事件

序号

事件 & 描述

1

newListener

event - 字符串,事件名称

listener - 处理事件函数

该事件在添加新监听器时被触发。

2

removeListener

event - 字符串,事件名称

listener - 处理事件函数

从指定监听器数组中删除一个监听器。需要注意的是,此操作将会改变处于被删监听器之后的那些监听器的索引。

main.js:

var events = require('events');
var eventEmitter = new events.EventEmitter();
// 监听器 #1
var listener1 = function listener1() {
   	console.log('监听器 listener1 执行。');
}
// 监听器 #2
var listener2 = function listener2() {
  	console.log('监听器 listener2 执行。');
}
// 绑定 connection 事件,处理函数为 listener1
eventEmitter.addListener('connection', listener1);
// 绑定 connection 事件,处理函数为 listener2
eventEmitter.on('connection', listener2);
var eventListeners = require('events').EventEmitter.listenerCount(eventEmitter,'connection');
console.log(eventListeners + " 个监听器监听连接事件。");
// 处理 connection 事件
eventEmitter.emit('connection');
// 移除监绑定的 listener1 函数
eventEmitter.removeListener('connection', listener1);
console.log("listener1 不再受监听。");
// 触发连接事件
eventEmitter.emit('connection');
eventListeners = require('events').EventEmitter.listenerCount(eventEmitter,'connection');
console.log(eventListeners + " 个监听器监听连接事件。");
console.log("程序执行完毕。");

运行方式:

node main.js

运行结果:

2 个监听器监听连接事件。

监听器 listener1 执行。

监听器 listener2 执行。

listener1 不再受监听。

监听器 listener2 执行。

1 个监听器监听连接事件。

程序执行完毕。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值