- 异步方法补丁
异步方法补丁
这里我们将看看如何通过异步回调传播zone的基本机制。 (注意:实际工作情况有点复杂,因为它们通过稍后讨论的任务调度机制进行调度,为了清楚起见,该示例进行了简化)。
// 保存原始的setTimeout引用let originalSetTimeout = window.setTimeout;// Overwrite the API with a function which wraps callback in zone.window.setTimeout = function(callback, delay) {// Invoke the original API but wrap the callback in zone.return originalSetTimeout(// Wrap the callback methodZone.current.wrap(callback),delay);}// Return a wrapped version of the callback which restores zone.Zone.prototype.wrap[c] = function(callback) {// Capture the current zonelet capturedZone = this;// Return a closure which executes the original closure in zone.return function() {// Invoke the original callback in the captured zone.return capturedZone.runGuarded(callback, this, arguments);};};
关键点:
- Zones 的猴子补丁方法只修补一次。
- 进入/离开 zone 只需更改Zone.current的值。(不需要进一步的猴子补丁)
- Zone.prototype.wrap method provides convenience for wrapping callbacks. (The wrapped callback is executed through Zone.prototype.runGuarded())
- Zone.prototype.runGuarded() is just like Zone.prototype.run(), but with extra try-catch block for handling exceptions described later.
// Save the original reference to Promise.prototype.then.let originalPromiseThen = Promise.prototype.then;// Overwrite the API with function which wraps the arguments// NOTE: this is simplified as actual API has more arguments.Promise.prototype.then = function(callback) {// Capture the current zonelet capturedZone = Zone.current;// Return a closure which executes the original closure in zone.function wrappedCallback() {return capturedZone.run(callback, this, arguments);};// Invoke the original callback in the captured zone.return originalPromiseThen.call(this, [wrappedCallback]);};
关键点:
- Promises handle their own exceptions and so they can’t use Zone.prototype.wrap(). (We could have separate API, but Promise is the exception to the rule, and so I did not feel justified creating its own API.)
- Promise API较多,在此示例中未显式显示。
