1

如何在不丢失其他侦听器的情况下侦听变量或对象属性上设置的值。

例如

var myModel;
myModel = 10;
var myListener1 = function(oldVal, newVal) {
    //do stuff
}
var myListener2 = function(oldVal, newVal) {
    //do other stuff
}

在这里,我希望在为变量 myModel 设置值时调用 myListener1 和 myListener2。稍后在其他函数中可能还希望在 setter 上向 myModel 添加另一个侦听器,因此它不应该覆盖现有的侦听器。

我知道应该有一种方法可以使用 Object.defineProperty()。此外,最好有一个针对 IE8+ 的跨浏览器解决方案。

4

2 回答 2

4

对于以下方法,您必须使用具有属性的对象,但它可以工作。


// this is very similar to using Object.watch()
// instead we attach multiple listeners
var myModel = (function () {
    var actualValue,
        interceptors = [];

    function callInterceptors(newValue) {
        for (var i = 0; i < interceptors.length; i += 1) {
            interceptors[i](newValue);
        }
    }

    return {
        get value() {
            // user never has access to the private variable "actualValue"
            // we can control what they get back from saying "myModel.value"
            return actualValue;
        },

        set value(newValue) {
            callInterceptors(newValue);
            actualValue = newValue;
        },

        listen : function (fn) {
            if (typeof fn === 'function') {
                interceptors.push(fn);
            }
        }
    };
}());

// add a listener
myModel.listen(function (passedValue) {
    alert('You tried to set myValue to ' + passedValue);
});

// add another listener
myModel.listen(function (passedValue) {
    alert('AAARRG! Why did you modify that value to ' + passedValue + '?!?!');
});

// our functions are called when we
// modify our value
myModel.value = 10;

jsFiddle 示例

于 2013-10-09T22:59:14.200 回答
0

仅通过函数更改 的值,myModel以便您可以在更改之后/之前运行侦听器函数。

var myModel = 10,
    myListener1 = function(oldVal, newVal) {
      // do stuff
    },
    myListener2 = function(oldVal, newVal) {
     // do other stuff
    },
    changeMyModel = function (value) {
      myListener1(myModel, value);
      myListener2(myModel, value);
      myModel = value;
      // do more stuff 
    };

   // changeMyModel(99) === myModel; true

或者

   var change = function (value) {
      myListener1(myModel, value);
      myListener2(myModel, value);
      return value;          
   };
   // myModel = change(45);
于 2013-10-09T02:22:30.927 回答