1

我正在尝试在 node.js 中创建一个模块/类来测量异步执行时间,但不明白它有什么问题。我创建了以下类“Measure.js”

var Measure = module.exports = function(param_timeout, param_cb) {

    this.timeout = param_timeout;
    this.cb = param_cb;
}

Measure.prototype = {
    startDate: "0",
    timeout:"0",
    cb:null,

    start : function() {
        this.startDate = new Date();
        console.log('started');
    },

    stop : function() {
        var stopDate = new Date();
        this.cb(null,(stopDate-this.startDate));
    }
}

我将它与以下代码一起使用:

var Measure = require('./Measure.js');
measure1 = new Measure(100,function(err,result){console.log('result: ' + result)});
measure1.start();
//do something 
measure1.stop();

它工作得很好。但是,如果我尝试这个:

var Measure = require('./Measure.js');
measure1 = new Measure(100,function(err,result){console.log('result: ' + result)});
measure1.start();
//do something 
setTimeout(measure1.stop,100);

它不起作用并引发 TypeError:

TypeError: Object #<Object> has no method 'cb'

我的代码有什么问题?

4

2 回答 2

6

当您直接调用对象的方法时,方法this内部引用您的对象,但当您尝试将其用作参数时,this将引用全局对象(globalwindow)。

在你的情况下最好更换

setTimeout(measure1.stop,100);

setTimeout(function() { measure1.stop(); }, 100);

有关this行为的更多信息:http: //bonsaiden.github.com/JavaScript-Garden/#function.this

于 2012-06-25T13:11:52.960 回答
3

可怕的 timeout-switch-context 错误再次出现!您会看到您所看到的,因为thissetTimeout 调用的函数中的对象不是measure1- 它是global(= window,当此脚本在浏览器中执行时)。引用MDN

执行的代码setTimeout()在与调用它的函数不同的执行上下文中运行。因此,this 被调用函数的关键字将设置为window(or global) 对象,它与被this调用函数的值不同setTimeout

顺便说一句,这很容易检查:

stop: function() {
  var stopDate = new Date();
  if (! this.cb) { 
    console.log(this); 
  } // *Window*/_display...
}

...并修复:

setTimeout(function() { 
  measure1.stop(); 
}, 100);
于 2012-06-25T13:14:58.747 回答