monkey patched: monkey patch指的是在运行时动态替换,一般是在startup的时候.
下面的测试代码可以让我们弄清楚,浏览器支持的原生函数setTimeout,是如何在Angular应用启动时被monkey patched的:
将setTimeout保存到一个Angular变量里。
在zone-evergreen.js的patchMethod方法里:
下图地2719行,能看到timers组下面的setTimeout,setInterval被monkey patched的入口:
Zone.__load_patch('timers', (global) => {
const set = 'set';
const clear = 'clear';
patchTimer(global, set, clear, 'Timeout');
patchTimer(global, set, clear, 'Interval');
patchTimer(global, set, clear, 'Immediate');
});
Zone.__load_patch('requestAnimationFrame', (global) => {
patchTimer(global, 'request', 'cancel', 'AnimationFrame');
patchTimer(global, 'mozRequest', 'mozCancel', 'AnimationFrame');
patchTimer(global, 'webkitRequest', 'webkitCancel', 'AnimationFrame');
});
Zone.__load_patch('blocking', (global, Zone) => {
const blockingMethods = ['alert', 'prompt', 'confirm'];
for (let i = 0; i < blockingMethods.length; i++) {
const name = blockingMethods[i];
patchMethod(global, name, (delegate, symbol, name) => {
return function (s, args) {
return Zone.current.run(delegate, global, args, name);
};
});
}
});
然后我在Angular应用里调用setTimeout时,执行的实际上是被monkey patched之后的版本,即代码1449行的函数体 patchDelegate, 一个委托函数:
更多Jerry的原创文章,尽在:“汪子熙”: