nodejs的事件处理机制以及事件环机制
ES6标准发布后,module成为标准,标准的使用是以export指令导出接口,以import引入模块,但是在我们一贯的node模块中,我们采用的是CommonJS规范,使用require引入模块,使用module.exports导出接口。
不把require和import整清楚,会在未来的标准编程中死的很难看。
require时代的模块
node编程中最重要的思想之一就是模块,而正是这个思想,让JavaScript的大规模工程成为可能。模块化编程在js界流行,也是基于此,随后在浏览器端,requirejs和seajs之类的工具包也出现了,可以说在对应规范下,require统治了ES6之前的所有模块化编程,即使现在,在ES6 module被完全实现之前,还是这样。
node的module遵循CommonJS规范,requirejs遵循AMD,seajs遵循CMD,虽各有不同,但总之还是希望保持较为统一的代码风格。
1.EventEmitter类
在Node.js中用于事件处理的event模块中,定义了一个EventEmitter类.所有可能触发的事件都是EventEmitter类子类的实例对象,EventEmitter类中的方法如下:
使用on方法绑定事件处理函数
var http = require('http');
var server = http.createServer();
server.on('request',function (req,res) {
console.log(req.url);
res.end('hello');
});
server.listen(1337,"127.0.0.1");
当我们请求1337端口的时候输出如下所示:
/ (注解:代表程序的根目录)
/favicon.ico (注解:代表页面在收藏夹中的显示图标)
在默认情况下,针对同一事件最多可以绑定10个事件处理函数:
server.setMaxListeners(10)
用once只执行一次:
server.once('request',function (req,res) {
console.log(req.url);
res.end('hello');
});
2.获取指定事件的事件处理函数的数量
var http = require('http');
var events = require('events');
var server = http.createServer();
server.on('request', function (req,res) {
if(req.url!='/favicon.ico'){
console.log('接收到客户端的请求');
}
});
server.on('request', function (req,res) {
if(req.url!='/favicon.ico'){
console.log(req.url);
}
res.end();
});
server.on('request', function (req,res) {
if(req.url!='/favicon.ico'){
console.log('发送响应完毕');
}
});
server.listen(1337,"127.0.0.1");
console.log(events.EventEmitter.listenerCount(server,'request'));
监听newListener和removeListener
var http = require('http');
var server = http.createServer();
var test = function () {
server.on('request',function (req,res) {
console.log("发送了");
});
}
server.on('removeListener',function (e,f) {
console.log("对"+e+"事件取消事件处理函数");
console.log(f);
});
server.on('newListener',function (e,f) {
console.log("对"+e+"事件添加事件处理函数");
console.log(f);
});
server.on('request', function (req,res) {
if(req.url!='/favicon.ico'){
console.log('接收到客户端的请求');
}
});
server.on('request', function (req,res) {
if(req.url!='/favicon.ico'){
console.log(req.url);
}
res.end();
});
server.on('request', function (req,res) {
if(req.url!='/favicon.ico'){
console.log('发送响应完毕');
}
});
server.on('request',test);
server.removeListener('request',test);
server.listen(1337,"127.0.0.1");
3.Node.js事件环机制
事件循环定义:当线程中的I/O任务完成之后就会执行指定的回调函数,并且将这个完成的事件放在事件队列的尾部,等待事件循环,当主线程再次循环到这个事件的时候,就会直接处理并且返回给上层调用,这个过程就是事件循环(Event Loop)。Node.js运行的原理图如下所示:
这个图是整个 Node.js 的运行原理,从左到右,从上到下,Node.js 被分为了四层,分别是 应用层、V8引擎层、Node API层和LIBUV层。
- 应用层:即 JavaScript 交互层,常见的就是 Node.js 的模块,比如 http,fs。
- V8引擎层:即利用 V8 引擎来解析JavaScript 语法,进而和下层 API 交互。
- Node API层:为上层模块提供系统调用,一般是由 C 语言来实现,和操作系统进行交互。
- LIBUV层:是跨平台的底层封装,实现了 事件循环、文件操作等,是 Node.js 实现异步的核心。
在Node.js的内部是通过线程池来完成I/O操作的,但是LIBUV层会针对不同的操作系统平台的差异性实现了统一调用,Node.js的单线程指的是JavaScript运行在单线程中,并不是说Node.js是单线程的,Node.js是一个多线程的平台,但是对于JavaScript的处理是单线程的。