Concept summary:

Node.js is single-threaded and handles only one event at a time during its event polling except for system IO. You can think of event polling as a large queue in which only one event is processed at each point in time. Even if your computer has multiple CPU cores, you can’t process multiple events in parallel. But this is what makes Node.js suitable for I/O applications, not CPU applications. In every I/O-type application, you just define a callback function for each input/output, and they are automatically added to the event polling processing queue. This callback function is triggered when the I/O operation is complete. The system then proceeds to process the other requests.

Process.nexttick () means to define an action and have it execute at the next event polling point in time

Function foo() {console.error('foo'); } process.nextTick(foo); console.error('bar'); Output: bar fooCopy the code

Usage scenarios

Cross-perform CPU-intensive tasks in multiple events

var http = require('http'); function compute() { // performs complicated calculations continuously // ... process.nextTick(compute); } http.createServer(function(req, res) { res.writeHead(200, {'Content-Type': 'text/plain'}); res.end('Hello World'); }), listen (5000, '127.0.0.1); compute();Copy the code

Maintain the principle that callbacks execute asynchronously

var client = net.connect(8124, function() { console.log('client connected'); client.write('world! \r\n'); }); In the above code, if for some reason net.connect() becomes synchronized, the callback is executed immediately, so the variables written to the client are never initialized. In this case we can use process.nexttick () to change asyncFake() to asyncFake() :  function asyncReal(data, callback) { process.nextTick(function() { callback(data === 'foo'); }); }Copy the code

Used during event triggering

For example, you want to write a library that reads data from a source file, and when it's finished, fires an event and passes the read data. You might write: var EventEmitter = require('events').EventEmitter; function StreamLibrary(resourceName) { this.emit('start'); // read from the file, and for every chunk read, do: this.emit('data', chunkRead); } StreamLibrary.prototype.__proto__ = EventEmitter.prototype; Inherit from EventEmitter var stream = new StreamLibrary('fooResource'); stream.on('start', function() { console.log('Reading has started'); }); stream.on('data', function(chunk) { console.log('Received: ' + chunk); }); However, the above code will never receive the "start" event, because the "start" event will be triggered immediately when the library is instantiated, but the event callback is not ready, so the client will not receive the event at all. Function StreamLibrary(resourceName) {var self = this; function StreamLibrary(resourceName) {var self = this; process.nextTick(function() { self.emit('start'); }); // read from the file, and for every chunk read, do: this.emit('data', chunkRead); }Copy the code