下面显示的代码以其当前形式编译和执行而没有错误——但是,一旦取消注释第 19 行上的“getNextAsyncTasks”类型声明,编译器就会崩溃并出现以下错误:
- 内部错误:无法获取属性“publicMembers”的值:对象为空或未定义
- 内部错误:IntelliSense 功能被禁用。尝试对源文件进行编辑以恢复有效的编译状态。
- 无法获取属性“publicMembers”的值:对象为空或未定义
- 命令 ""C:\Program Files (x86)\Microsoft SDKs\TypeScript\0.8.0.0\tsc" ... " 以代码 1 退出。
虽然我知道如何在这种特殊情况下避免它,但当我在代码库的其他区域工作时,错误会一直默默地发生。调试可能非常困难,因为当我注意到它发生时,我经常进行大量更改,并且没有暗示这次问题可能来自哪里。
理想情况下,如果有人能让我知道导致此崩溃发生的原因,我将不胜感激,这样我就可以更好地了解如何避免它/解决它。
module abstract {
export interface ICallback {
(...args: any[]): any;
}
export interface IAsyncOp {
(callback: ICallback): any;
}
}
export class Chain {
task : abstract.IAsyncOp;
constructor(...tasks : abstract.IAsyncOp[]) {
if (tasks.length === 1) this.task = tasks[0];
else if (tasks.length > 1) this.task = cb => this.sync(() => cb.apply(null, arguments), tasks);
else this.task = cb => cb();
}
next(getNextAsyncTasks /*: (...args: any[]) => abstract.IAsyncOp*/) {
return new Chain(
(cb: abstract.ICallback) =>
this.task(
() => getNextAsyncTasks.apply(null, arguments)(cb);
)
);
}
last(cb : abstract.ICallback) {
this.task(cb);
}
sync(cb: (...resultArgs: IArguments[]) => void, tasks: abstract.IAsyncOp[]) {
var resultArgs : IArguments[] = [], done = 0;
var getCb = (i) => {
return () => {
resultArgs[i] = arguments;
done++;
if (done === tasks.length) cb.apply(null, resultArgs);
}
};
for (var i = 0, op; op = tasks[i]; i++) op(getCb(i));
}
}
// use example
new Chain( cb => setTimeout(() => cb("foo"), 60)
, cb => setTimeout(() => cb("bar"), 10)
).next((op1, op2) => { console.log(op1, op2)
; return cb => setTimeout(() => cb(op1[0], op2[0]), 120)
}
).next((foo, bar) => { console.log(foo, bar)
; return cb => setTimeout(() => cb(foo, bar, "baz"), 30)
}
).last((foo, bar, baz) => console.log(foo, bar, baz));
编辑:更新了“下一个”,所以参数的类型签名总是正确的。