在Node.js中,JavaScript代码运行在V8引擎上。由于JavaScript是单线程语言,一次只能处理一个事件。为了解决这个问题,Node.js引入了事件驱动模型。每个进行IO操作的函数都是异步的,当这个函数被调用的时候,它不会立即执行IO操作,而是将操作封装为一个任务,加入到事件队列中。之后函数立即返回,处理下一个任务。
事件循环(Event Loop)就像是一个循环,不停地从事件队列中取出任务执行。当事件队列中没有任务时,事件循环会等待新的IO任务入队。当IO操作完成,相关的回调函数被添加到任务队列,等待事件循环取出执行。
通过这种方式,Node.js实现了非阻塞IO模型。当IO操作正在进行,Node.js不会空闲等待结果,而是继续处理其他任务。待IO操作完成,回调函数入队,才会被执行。这就大大提升了程序的并发性能,使得Node.js可以有效地处理大量并发请求。
以下是一个Node.js的非阻塞IO的代码示例。这个代码通过Node.js的fs
模块,异步读取一个文本文件。当读取操作完成后,执行回调函数。
const fs = require('fs');
fs.readFile('myfile.txt', 'utf8' , (err, data) => {
if (err) {
console.error(err);
return;
}
console.log(data);
});
console.log('Reading file...');
这里,fs.readFile
函数的参数包括文件名,编码,以及一个回调函数。回调函数在读取文件完成后被调用。err参数对应错误信息,如果读取成功,err为null。data参数对应文件内容。
值得注意的是,虽然读取文件是一个耗时的IO操作,但由于Node.js的非阻塞特性,fs.readFile
函数并不会阻塞程序,而是拿到任务后立即返回。因此,'Reading file...'这行代码会先于回调函数内的console.log(data)
执行,即使文件读取未完成。