在javascript中取消回调

时间:2015-04-02 00:38:56

标签: javascript function callback

我有类似的东西:

function MyObject() {
  var self = this;

  this.callback = function() {
    self.finishParams = Array.prototype.slice.call(arguments);
    self.parent.finish();
  }

  this.start = function() {
    this.currentCallback = this.callback
    this.startFunc.apply(this.startFunc, this.startParams.concat(this.currentCallback));
  }
}

this.startFunc是一个类似函数的函数(param1,param2,param3,callback) 我无法控制this.startFunc,除了它会用一些参数调用回调。

问题

我有一个this.currentCallback,因为我需要能够取消回调。也就是说,我已经调用了this.startFunc并且需要阻止回调。

问题是,MyObject可能会发送另一个回调(一次不会发送2个回调),但如果我不需要立即取消第一个回复,我就不知道哪个回复有效让他们回来!可能会让人感到困惑,所以这里有一个图表:

  • 发送回拨1关闭
  • 需要取消!取消回调在某种程度上
  • 发送回拨2关闭(仍然说功能有回调1)

到此为止,如果我没有取消A,那么当我收到回调时,我不知道它是什么。如果我取消A,那么我知道它的B,没有人担心。

如果你不明白,请告诉我:)。

1 个答案:

答案 0 :(得分:1)

评论中提出的方案概念验证:为每个回调创建一个新的闭包,让回调识别它是否有效。

function foreignAPIThatStartsACallback(callback) {
  setTimeout(callback, 1000);
}

var activeCallback;

function wrapCallback(callback) {
  var args = Array.prototype.slice.call(arguments, 1);
  var that = this;
  var wrappedCallback = function() {
    if (wrappedCallback == activeCallback) {
      callback.apply(that, args);
    }
  }
  activeCallback = wrappedCallback;
  return wrappedCallback;
}

function myCallback(what, who) {
  console.log(who + " says " + what);
}

foreignAPIThatStartsACallback(wrapCallback(myCallback, "Hello", "Mario"));
foreignAPIThatStartsACallback(wrapCallback(myCallback, "Goodbye", "Luigi"));
// Mario is cancelled when Luigi gets started

有多种可能的活动:

function foreignAPIThatStartsACallback(callback) {
  setTimeout(callback, 1000);
}

var activeCallbacks = {};

function wrapCallback(callback) {
  if (!wrapCallback.count) wrapCallback.count = 0;
  wrapCallback.count++;
  var args = Array.prototype.slice.call(arguments, 1);
  var that = this;
  var wrappedCallback = function() {
    if (wrappedCallback.id in activeCallbacks) {
      cancelCallback(wrappedCallback);
      callback.apply(that, args);
    }
  }
  wrappedCallback.id = wrapCallback.count;
  activeCallbacks[wrapCallback.count] = true;
  return wrappedCallback;
}

function cancelCallback(wrappedCallback) {
  delete activeCallbacks[wrappedCallback.id];
}

function myCallback(what, who) {
  console.log(who + " says " + what);
}


var marioCallback = wrapCallback(myCallback, "Hello", "Mario");
foreignAPIThatStartsACallback(marioCallback);
var luigiCallback = wrapCallback(myCallback, "Goodbye", "Luigi");
foreignAPIThatStartsACallback(luigiCallback);
var daisyCallback = wrapCallback(myCallback, "Mama?", "Peach");
foreignAPIThatStartsACallback(daisyCallback);

cancelCallback(luigiCallback);
// Mario and Daisy go off