197

在 JavaScript 中,使用 bind() 删除作为事件侦听器添加的函数的最佳方法是什么?

例子

(function(){

    // constructor
    MyClass = function() {
        this.myButton = document.getElementById("myButtonID");
        this.myButton.addEventListener("click", this.clickListener.bind(this));
    };

    MyClass.prototype.clickListener = function(event) {
        console.log(this); // must be MyClass
    };

    // public method
    MyClass.prototype.disableButton = function() {
        this.myButton.removeEventListener("click", ___________);
    };

})();

我能想到的唯一方法是跟踪每个添加了绑定的侦听器。

上面这个方法的例子:

(function(){

    // constructor
    MyClass = function() {
        this.myButton = document.getElementById("myButtonID");
        this.clickListenerBind = this.clickListener.bind(this);
        this.myButton.addEventListener("click", this.clickListenerBind);
    };

    MyClass.prototype.clickListener = function(event) {
        console.log(this); // must be MyClass
    };

    // public method
    MyClass.prototype.disableButton = function() {
        this.myButton.removeEventListener("click", this.clickListenerBind);
    };

})();

有没有更好的方法来做到这一点?

4

10 回答 10

320

尽管@machineghost 说的是真的,添加和删除事件的方式相同,但等式中缺少的部分是:

调用后会创建一个新的函数引用.bind()

请参阅bind() 是否更改函数引用?| 如何永久设置?

因此,要添加或删除它,请将引用分配给变量:

var x = this.myListener.bind(this);
Toolbox.addListener(window, 'scroll', x);
Toolbox.removeListener(window, 'scroll', x);

这对我来说按预期工作。

于 2014-04-04T18:46:21.590 回答
52

对于那些在向/从 Flux 存储注册/删除 React 组件的侦听器时遇到此问题的人,请将以下行添加到组件的构造函数中:

class App extends React.Component {
  constructor(props){
    super(props);
    // it's a trick! needed in order to overcome the remove event listener
    this.onChange = this.onChange.bind(this);  
  }
  // then as regular...
  componentDidMount (){
    AppStore.addChangeListener(this.onChange);
  }
  
  componentWillUnmount (){
    AppStore.removeChangeListener(this.onChange);
  }

  onChange () {
    let state = AppStore.getState();
    this.setState(state);
  }
  
  render() {
    // ...
  }
  
}

于 2015-10-28T08:39:35.063 回答
2

是否使用绑定函数都没有关系;您删除它的方式与任何其他事件处理程序相同。如果您的问题是绑定版本是它自己的独特功能,您可以跟踪绑定版本,或者使用removeEventListener不采用特定处理程序的签名(尽管这当然会删除相同类型的其他事件处理程序)。

(附带说明一下,addEventListener并非在所有浏览器中都有效;您确实应该使用像 jQuery 这样的库来为您以跨浏览器的方式进行事件连接。此外,jQuery 具有命名空间事件的概念,它允许你绑定到“click.foo”;当你想删除事件时,你可以告诉 jQuery“删除所有 foo 事件”,而不必知道特定的处理程序或删除其他处理程序。)

于 2012-07-19T16:44:01.273 回答
1

jQuery解决方案:

let object = new ClassName();
let $elem = $('selector');

$elem.on('click', $.proxy(object.method, object));

$elem.off('click', $.proxy(object.method, object));
于 2018-03-30T10:23:04.337 回答
1

我们在无法更改的库中遇到了这个问题。Office Fabric UI,这意味着我们无法更改添加事件处理程序的方式。我们解决它的方法是覆盖原型addEventListener上的。EventTarget

这将在对象上添加一个新功能element.removeAllEventListers("click")

(原帖:从结构对话框覆盖中删除点击处理程序

        <script>
            (function () {
                "use strict";

                var f = EventTarget.prototype.addEventListener;

                EventTarget.prototype.addEventListener = function (type, fn, capture) {
                    this.f = f;
                    this._eventHandlers = this._eventHandlers || {};
                    this._eventHandlers[type] = this._eventHandlers[type] || [];
                    this._eventHandlers[type].push([fn, capture]);
                    this.f(type, fn, capture);
                }

                EventTarget.prototype.removeAllEventListeners = function (type) {
                    this._eventHandlers = this._eventHandlers || {};
                    if (type in this._eventHandlers) {
                        var eventHandlers = this._eventHandlers[type];
                        for (var i = eventHandlers.length; i--;) {
                            var handler = eventHandlers[i];
                            this.removeEventListener(type, handler[0], handler[1]);
                        }
                    }
                }

                EventTarget.prototype.getAllEventListeners = function (type) {
                    this._eventHandlers = this._eventHandlers || {};
                    this._eventHandlers[type] = this._eventHandlers[type] || [];
                    return this._eventHandlers[type];
                }

            })();
        </script>
于 2019-04-03T09:03:26.627 回答
0

这是解决方案:

var o = {
  list: [1, 2, 3, 4],
  add: function () {
    var b = document.getElementsByTagName('body')[0];
    b.addEventListener('click', this._onClick());

  },
  remove: function () {
    var b = document.getElementsByTagName('body')[0];
    b.removeEventListener('click', this._onClick());
  },
  _onClick: function () {
    this.clickFn = this.clickFn || this._showLog.bind(this);
    return this.clickFn;
  },
  _showLog: function (e) {
    console.log('click', this.list, e);
  }
};


// Example to test the solution
o.add();

setTimeout(function () {
  console.log('setTimeout');
  o.remove();
}, 5000);
于 2016-10-28T20:14:28.267 回答
0

正如其他人所说,bind创建一个新的函数实例,因此除非以某种方式记录,否则无法删除事件侦听器。

为了获得更漂亮的代码风格,您可以将方法函数设置为惰性 getter,以便在第一次访问时自动替换为绑定版本:

class MyClass {
  activate() {
    window.addEventListener('click', this.onClick);
  }

  deactivate() {
    window.removeEventListener('click', this.onClick);
  }

  get onClick() {
    const func = (event) => {
      console.log('click', event, this);
    };
    Object.defineProperty(this, 'onClick', {value: func});
    return func;
  }
}

如果不支持 ES6 箭头功能,请const func = (function(event){...}).bind(this)改用。

Raichman Sergey 的方法也很好,尤其是在课堂上。这种方法的优点是它更加自我完成,并且在其他地方没有单独的代码。它也适用于没有构造函数或启动器的对象。

于 2022-03-01T12:28:12.700 回答
-1

如果你想使用'onclick',如上所述,你可以试试这个:

(function(){
    var singleton = {};

    singleton = new function() {
        this.myButton = document.getElementById("myButtonID");

        this.myButton.onclick = function() {
            singleton.clickListener();
        };
    }

    singleton.clickListener = function() {
        console.log(this); // I also know who I am
    };

    // public function
    singleton.disableButton = function() {
        this.myButton.onclick = "";
    };
})();

我希望它有所帮助。

于 2012-07-19T17:28:16.130 回答
-1

可以使用关于 ES7:

class App extends React.Component {
  constructor(props){
    super(props);
  }
  componentDidMount (){
    AppStore.addChangeListener(this.onChange);
  }

  componentWillUnmount (){
    AppStore.removeChangeListener(this.onChange);
  }

  onChange = () => {
    let state = AppStore.getState();
    this.setState(state);
  }

  render() {
    // ...
  }

}
于 2019-08-05T07:06:57.927 回答
-2

已经有一段时间了,但 MDN 对此有一个超级解释。这比这里的东西对我的帮助更大。

MDN :: EventTarget.addEventListener - 处理程序中“this”的值

它为handleEvent 函数提供了一个很好的替代方案。

这是一个有和没有绑定的例子:

var Something = function(element) {
  this.name = 'Something Good';
  this.onclick1 = function(event) {
    console.log(this.name); // undefined, as this is the element
  };
  this.onclick2 = function(event) {
    console.log(this.name); // 'Something Good', as this is the binded Something object
  };
  element.addEventListener('click', this.onclick1, false);
  element.addEventListener('click', this.onclick2.bind(this), false); // Trick
}

上面示例中的一个问题是您无法使用 bind 删除侦听器。另一种解决方案是使用一个名为 handleEvent 的特殊函数来捕获任何事件:

于 2014-05-01T06:30:08.617 回答