浅谈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) 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) |
7 | listeners(event) |
8 | emit(event, [arg1], [arg2], […]) |
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 个监听器监听连接事件。
程序执行完毕。
还没有评论,来说两句吧...