timers: implement setImmediate
This commit is contained in:
parent
d15bfc04cd
commit
382f22f229
@ -46,3 +46,20 @@ event loop performance -- use wisely.
|
|||||||
If you had previously `unref()`d a timer you can call `ref()` to explicitly
|
If you had previously `unref()`d a timer you can call `ref()` to explicitly
|
||||||
request the timer hold the program open. If the timer is already `ref`d calling
|
request the timer hold the program open. If the timer is already `ref`d calling
|
||||||
`ref` again will have no effect.
|
`ref` again will have no effect.
|
||||||
|
|
||||||
|
## setImmediate(callback, [arg], [...])
|
||||||
|
|
||||||
|
To schedule the "immediate" execution of `callback`. Returns an `immediateId`
|
||||||
|
for possible use with `clearImmediate()`. Optionally you can also pass
|
||||||
|
arguments to the callback.
|
||||||
|
|
||||||
|
Immediates are queued in the order created, and are popped off the queue once
|
||||||
|
per loop iteration. This is different from `process.nextTick` which will
|
||||||
|
execute `process.maxTickDepth` queued callbacks per iteration. `setImmediate`
|
||||||
|
will yield to the event loop after firing a queued callback to make sure I/O is
|
||||||
|
not being starved. While order is preserved for execution, other I/O events may
|
||||||
|
fire between any two scheduled immediate callbacks.
|
||||||
|
|
||||||
|
## clearImmediate(immediateId)
|
||||||
|
|
||||||
|
Stops an immediate from triggering.
|
||||||
|
@ -277,3 +277,74 @@ Timeout.prototype.close = function() {
|
|||||||
exports.unenroll(this);
|
exports.unenroll(this);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
var immediateTimer = null;
|
||||||
|
var immediateQueue = { started: false };
|
||||||
|
L.init(immediateQueue);
|
||||||
|
|
||||||
|
|
||||||
|
function lazyImmediateInit() { // what's in a name?
|
||||||
|
if (immediateTimer) return;
|
||||||
|
immediateTimer = new Timer;
|
||||||
|
immediateTimer.ontimeout = processImmediate;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function processImmediate() {
|
||||||
|
var immediate;
|
||||||
|
if (L.isEmpty(immediateQueue)) {
|
||||||
|
immediateTimer.stop();
|
||||||
|
immediateQueue.started = false;
|
||||||
|
} else {
|
||||||
|
immediate = L.shift(immediateQueue);
|
||||||
|
|
||||||
|
if (immediate.domain) immediate.domain.enter();
|
||||||
|
|
||||||
|
immediate._onTimeout();
|
||||||
|
|
||||||
|
if (immediate.domain) immediate.domain.exit();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
exports.setImmediate = function(callback) {
|
||||||
|
var immediate = {}, args;
|
||||||
|
|
||||||
|
L.init(immediate);
|
||||||
|
|
||||||
|
immediate._onTimeout = callback;
|
||||||
|
|
||||||
|
if (arguments.length > 1) {
|
||||||
|
args = Array.prototype.slice.call(arguments, 1);
|
||||||
|
immediate._onTimeout = function() {
|
||||||
|
callback.apply(null, args);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!immediateQueue.started) {
|
||||||
|
lazyImmediateInit();
|
||||||
|
immediateTimer.start(0, 1);
|
||||||
|
immediateQueue.started = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (process.domain) immediate.domain = process.domain;
|
||||||
|
|
||||||
|
L.append(immediateQueue, immediate);
|
||||||
|
|
||||||
|
return immediate;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
exports.clearImmediate = function(immediate) {
|
||||||
|
if (!immediate) return;
|
||||||
|
|
||||||
|
immediate._onTimeout = undefined;
|
||||||
|
|
||||||
|
L.remove(immediate);
|
||||||
|
|
||||||
|
if (L.isEmpty(immediateQueue)) {
|
||||||
|
immediateTimer.stop();
|
||||||
|
immediateQueue.started = false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
10
src/node.js
10
src/node.js
@ -184,6 +184,16 @@
|
|||||||
var t = NativeModule.require('timers');
|
var t = NativeModule.require('timers');
|
||||||
return t.clearInterval.apply(this, arguments);
|
return t.clearInterval.apply(this, arguments);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
global.setImmediate = function() {
|
||||||
|
var t = NativeModule.require('timers');
|
||||||
|
return t.setImmediate.apply(this, arguments);
|
||||||
|
};
|
||||||
|
|
||||||
|
global.clearImmediate = function() {
|
||||||
|
var t = NativeModule.require('timers');
|
||||||
|
return t.clearImmediate.apply(this, arguments);
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
startup.globalConsole = function() {
|
startup.globalConsole = function() {
|
||||||
|
@ -81,8 +81,10 @@ process.on('exit', function() {
|
|||||||
if (!exports.globalCheck) return;
|
if (!exports.globalCheck) return;
|
||||||
var knownGlobals = [setTimeout,
|
var knownGlobals = [setTimeout,
|
||||||
setInterval,
|
setInterval,
|
||||||
|
setImmediate,
|
||||||
clearTimeout,
|
clearTimeout,
|
||||||
clearInterval,
|
clearInterval,
|
||||||
|
clearImmediate,
|
||||||
console,
|
console,
|
||||||
Buffer,
|
Buffer,
|
||||||
process,
|
process,
|
||||||
|
53
test/simple/test-timers-immediate.js
Normal file
53
test/simple/test-timers-immediate.js
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
// Copyright Joyent, Inc. and other Node contributors.
|
||||||
|
//
|
||||||
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
||||||
|
// copy of this software and associated documentation files (the
|
||||||
|
// "Software"), to deal in the Software without restriction, including
|
||||||
|
// without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
||||||
|
// persons to whom the Software is furnished to do so, subject to the
|
||||||
|
// following conditions:
|
||||||
|
//
|
||||||
|
// The above copyright notice and this permission notice shall be included
|
||||||
|
// in all copies or substantial portions of the Software.
|
||||||
|
//
|
||||||
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
||||||
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
||||||
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
||||||
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
||||||
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
||||||
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
|
var common = require('../common');
|
||||||
|
var assert = require('assert');
|
||||||
|
|
||||||
|
var immediateA = false,
|
||||||
|
immediateB = false,
|
||||||
|
immediateC = [],
|
||||||
|
before;
|
||||||
|
|
||||||
|
setImmediate(function() {
|
||||||
|
try {
|
||||||
|
immediateA = process.hrtime(before);
|
||||||
|
} catch(e) {
|
||||||
|
console.log('failed to get hrtime with offset');
|
||||||
|
}
|
||||||
|
clearImmediate(immediateB);
|
||||||
|
});
|
||||||
|
|
||||||
|
before = process.hrtime();
|
||||||
|
|
||||||
|
immediateB = setImmediate(function() {
|
||||||
|
immediateB = true;
|
||||||
|
});
|
||||||
|
|
||||||
|
setImmediate(function(x, y, z) {
|
||||||
|
immediateC = [x, y, z];
|
||||||
|
}, 1, 2, 3);
|
||||||
|
|
||||||
|
process.on('exit', function() {
|
||||||
|
assert.ok(immediateA, 'Immediate should happen after normal execution');
|
||||||
|
assert.notStrictEqual(immediateB, true, 'immediateB should not fire');
|
||||||
|
assert.deepEqual(immediateC, [1, 2, 3], 'immediateC args should match');
|
||||||
|
});
|
Loading…
x
Reference in New Issue
Block a user