Прослушивание переменных изменений в JavaScript

Возможно ли иметь событие в JS, которое срабатывает при изменении значения определенной переменной? JQuery принимается.

Ответ 2

Да, теперь это возможно!

Я знаю, что это старый поток, но теперь этот эффект возможен с помощью аксессуаров (getters и seters): https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Working_with_Objects#Defining_getters_and_setters

Вы можете определить такой объект, в котором aInternal представляет поле a:

x = {
  aInternal: 10,
  aListener: function(val) {},
  set a(val) {
    this.aInternal = val;
    this.aListener(val);
  },
  get a() {
    return this.aInternal;
  },
  registerListener: function(listener) {
    this.aListener = listener;
  }
}

Затем вы можете зарегистрировать прослушиватель, используя следующее:

x.registerListener(function(val) {
  alert("Someone changed the value of x.a to " + val);
});

Поэтому всякий раз, когда что-либо меняет значение x.a, функция слушателя будет запущена. Выполнение следующей строки приведет к появлению предупреждения:

x.a = 42;

См. пример здесь: https://jsfiddle.net/5o1wf1bn/1/

Вы также можете использовать массив прослушивателей вместо одного слота прослушивателя, но я хотел бы дать вам самый простой пример.

Ответ 3

Большинство ответов на этот вопрос либо устарели, либо неэффективны, либо требуют включения больших раздутых библиотек:

  • Object.watch и Object.observe устарели и не должны использоваться.
  • onPropertyChange - это обработчик событий элемента DOM, который работает только в некоторых версиях IE.
  • Object.defineProperty позволяет сделать свойство объекта неизменным, что позволит вам обнаруживать предпринятые изменения, но также будет блокировать любые изменения.
  • Определение сеттеров и геттеров работает, но требует большого количества установочного кода и не очень хорошо работает, когда вам нужно удалить или создать новые свойства.

Сегодня вы можете использовать объект Proxy для мониторинга (и перехвата) изменений, внесенных в объект. Он специально создан для того, что пытается сделать ОП. Вот основной пример:

var targetObj = {};
var targetProxy = new Proxy(targetObj, {
  set: function (target, key, value) {
      console.log('${key} set to ${value}');
      target[key] = value;
      return true;
  }
});

targetProxy.hello_world = "test"; // console: 'hello_world set to test'

Единственными недостатками объекта Proxy являются:

  1. Объект Proxy недоступен в более старых браузерах (например, IE11), и полизаполнение не может полностью копировать функции Proxy.
  2. Прокси-объекты не всегда ведут себя должным образом со специальными объектами (например, Date) - Proxy объект лучше всего сочетать с простыми объектами или массивами.

Если вам нужно наблюдать за изменениями, внесенными во вложенный объект, вам нужно использовать специализированную библиотеку, такую как Observable Slim (которую я опубликовал), которая работает следующим образом:

var test = {testing:{}};
var p = ObservableSlim.create(test, true, function(changes) {
    console.log(JSON.stringify(changes));
});

p.testing.blah = 42; // console:  [{"type":"add","target":{"blah":42},"property":"blah","newValue":42,"currentPath":"testing.blah",jsonPointer:"/testing/blah","proxy":{"blah":42}}]

Ответ 4

Нет.

Но если это действительно так важно, у вас есть 2 варианта (сначала проверено, а второе нет):

Во-первых, используйте сеттеры и геттеры, например:

var myobj = {a : 1};

function create_gets_sets(obj) { // make this a framework/global function
    var proxy = {}
    for ( var i in obj ) {
        if (obj.hasOwnProperty(i)) {
            var k = i;
            proxy["set_"+i] = function (val) { this[k] = val; };
            proxy["get_"+i] = function ()    { return this[k]; };
        }
    }
    for (var i in proxy) {
        if (proxy.hasOwnProperty(i)) {
            obj[i] = proxy[i];
        }
    }
}

create_gets_sets(myobj);

то вы можете сделать что-то вроде:

function listen_to(obj, prop, handler) {
    var current_setter = obj["set_" + prop];
    var old_val = obj["get_" + prop]();
    obj["set_" + prop] = function(val) { current_setter.apply(obj, [old_val, val]); handler(val));
}

затем установите слушатель как:

listen_to(myobj, "a", function(oldval, newval) {
    alert("old : " + oldval + " new : " + newval);
}

Во-вторых, я действительно забыл, я отправлю, пока я думаю об этом:)

EDIT: О, я помню:) Вы можете поставить часы на значение:

Учитывая myobj выше, с 'a' на нем:

function watch(obj, prop, handler) { // make this a framework/global function
    var currval = obj[prop];
    function callback() {
        if (obj[prop] != currval) {
            var temp = currval;
            currval = obj[prop];
            handler(temp, currval);
        }
    }
    return callback;
}

var myhandler = function (oldval, newval) {
    //do something
};

var intervalH = setInterval(watch(myobj, "a", myhandler), 100);

myobj.set_a(2);

Ответ 5

Использование Prototype: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/defineProperty

// Console
function print(t) {
  var c = document.getElementById('console');
  c.innerHTML = c.innerHTML + '<br />' + t;
}

// Demo
var myVar = 123;

Object.defineProperty(this, 'varWatch', {
  get: function () { return myVar; },
  set: function (v) {
    myVar = v;
    print('Value changed! New value: ' + v);
  }
});

print(varWatch);
varWatch = 456;
print(varWatch);
<pre id="console">
</pre>

Ответ 6

Извините, что вызвал старую нить, но вот небольшое руководство для тех, кто (как и я!) не видит, как работает Eli Grey:

var test = new Object();
test.watch("elem", function(prop,oldval,newval){
    //Your code
    return newval;
});

Надеюсь, это поможет кому-то

Ответ 7

Как Ответ Люка Шафера (note): это относится к его оригинальному сообщению, но вся суть здесь остается в силе после редактирования), Я также предложил пару методов Get/Set для доступа к вашему значению.

Однако я бы предложил некоторые изменения (и почему я отправляю...).

Проблема с этим кодом заключается в том, что поле a объекта myobj доступно напрямую, поэтому можно получить доступ к нему/изменить его значение без запуска слушателей:

var myobj = { a : 5, get_a : function() { return this.a;}, set_a : function(val) { this.a = val; }}
/* add listeners ... */
myobj.a = 10; // no listeners called!

Инкапсуляция

Итак, чтобы гарантировать, что слушатели действительно вызваны, нам придется запретить прямой доступ к полю a. Как это сделать? Используйте закрытие!

var myobj = (function() { // Anonymous function to create scope.

    var a = 5;            // 'a' is local to this function
                          // and cannot be directly accessed from outside
                          // this anonymous function scope

    return {
        get_a : function() { return a; },   // These functions are closures:
        set_a : function(val) { a = val; }  // they keep reference to
                                            // something ('a') that was on scope
                                            // where they were defined
    };
})();

Теперь вы можете использовать тот же метод для создания и добавления слушателей, как предложил Люк, но вы можете быть уверены, что нет возможности читать или записывать на a незаметно!

Программный ввод инкапсулированных полей

Находясь на дорожке Luke, я предлагаю простой способ добавить инкапсулированные поля и соответствующие геттеры/сеттеры к объектам с помощью простого вызова функции.

Обратите внимание, что это будет работать только с типами значений. Для этого, чтобы работать со ссылочными типами, необходимо было бы выполнить некоторую глубокую копию (см. этот.

function addProperty(obj, name, initial) {
    var field = initial;
    obj["get_" + name] = function() { return field; }
    obj["set_" + name] = function(val) { field = val; }
}

Это работает так же, как и раньше: мы создаем локальную переменную в функции, а затем создаем закрытие.

Как его использовать? Простой:

var myobj = {};
addProperty(myobj, "total", 0);
window.alert(myobj.get_total() == 0);
myobj.set_total(10);
window.alert(myobj.get_total() == 10);

Ответ 8

Если вы используете jQuery {UI} (который каждый должен использовать:-)), вы можете использовать .change() со скрытым < input/ > элемент.

Ответ 9

AngularJS (Я знаю, что это не JQuery, но это может помочь. [Pure JS хорош только в теории]):

$scope.$watch('data', function(newValue) { ..

где "данные" - это имя вашей переменной в области.

Существует ссылка на doc.

Ответ 10

Для тех, кто настроен через пару лет:

Доступно решение для большинства браузеров (и IE6 +), которое использует событие onpropertychange и более новую спецификацию specProperty. Небольшой улов состоит в том, что вам нужно сделать вашу переменную dom-объектом.

Полная информация:

http://johndyer.name/native-browser-get-set-properties-in-javascript/

Ответ 11

Функциональность, которую вы ищете, может быть достигнута с помощью метода defineProperty(), который доступен только для современных браузеров:

https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/defineProperty

Я написал расширение jQuery, которое имеет некоторую аналогичную функциональность, если вам нужна более кросс-браузерная поддержка:

https://github.com/jarederaj/jQueue

Небольшое расширение jQuery, которое обрабатывает обратные вызовы в очереди существование переменной, объекта или ключа. Вы можете назначить любое количество обратные вызовы для любого количества точек данных, на которые может процессы, выполняющиеся в фоновом режиме. jQueue слушает и ждет эти данные, которые вы указываете, должны появиться, а затем правильный ответ с его аргументами.

Ответ 12

Не напрямую: вам нужен пара getter/setter с каким-то интерфейсом addListener/removeListener... или плагин NPAPI (но эта другая история вообще).

Ответ 13

//ex:
/*
var x1 = {currentStatus:undefined};
your need is x1.currentStatus value is change trigger event ?
below the code is use try it.
*/
function statusChange(){
    console.log("x1.currentStatus_value_is_changed"+x1.eventCurrentStatus);
};

var x1 = {
    eventCurrentStatus:undefined,
    get currentStatus(){
        return this.eventCurrentStatus;
    },
    set currentStatus(val){
        this.eventCurrentStatus=val;
      //your function();
    }
};

или

/*  var x1 = {
eventCurrentStatus:undefined,
currentStatus : {
    get : function(){
        return Events.eventCurrentStatus
        },
    set : function(status){
        Events.eventCurrentStatus=status;

    },
}*/
console.log("eventCurrentStatus = "+ x1.eventCurrentStatus);
x1.currentStatus="create"
console.log("eventCurrentStatus = "+ x1.eventCurrentStatus);
x1.currentStatus="edit"
console.log("eventCurrentStatus = "+ x1.eventCurrentStatus);
console.log("currentStatus = "+ x1.currentStatus);

или

/* global variable ku*/
    var jsVarEvents={};
    Object.defineProperty(window, "globalvar1", {//no i18n
        get: function() { return window.jsVarEvents.globalvarTemp},
        set: function(value) { window.window.jsVarEvents.globalvarTemp = value; }
    });
    console.log(globalvar1);
    globalvar1=1;
    console.log(globalvar1);

Ответ 14

Довольно простое и упрощенное решение - просто использовать вызов функции для установки значения глобальной переменной и никогда не устанавливать его значение напрямую. Таким образом, у вас есть полный контроль:

var globalVar;

function setGlobalVar(value) {
    globalVar = value;
    console.log("Value of globalVar set to: " + globalVar);
    //Whatever else
}

Нет способа обеспечить это, он просто требует программирования дисциплины... хотя вы можете использовать grep (или что-то подобное), чтобы проверить, что нигде ваш код не устанавливает значение globalVar.

Или вы можете инкапсулировать его в методах создания объектов и пользователей, а также в простое мышление.

Ответ 15

Это невозможно.

Однако это можно сделать с помощью CustomEvent: https://developer.mozilla.org/en-US/docs/Web/API/CustomEvent/CustomEvent

Метод ниже принимает массив имен переменных в качестве входных данных и добавляет прослушиватель событий для каждой переменной и запускает событие для любых изменений значения переменных.

Метод использует опрос для обнаружения изменения значения. Вы можете увеличить значение таймаута в миллисекундах.

function watchVariable(varsToWatch) {
    let timeout = 1000;
    let localCopyForVars = {};
    let pollForChange = function () {
        for (let varToWatch of varsToWatch) {
            if (localCopyForVars[varToWatch] !== window[varToWatch]) {
                let event = new CustomEvent('onVar_' + varToWatch + 'Change', {
                    detail: {
                        name: varToWatch,
                        oldValue: localCopyForVars[varToWatch],
                        newValue: window[varToWatch]
                    }
                });
                document.dispatchEvent(event);
                localCopyForVars[varToWatch] = window[varToWatch];
            }
        }
        setTimeout(pollForChange, timeout);
    };
    let respondToNewValue = function (varData) {
        console.log("The value of the variable " + varData.name + " has been Changed from " + varData.oldValue + " to " + varData.newValue + "!!!"); 
    }
    for (let varToWatch of varsToWatch) {
        localCopyForVars[varToWatch] = window[varToWatch];
        document.addEventListener('onVar_' + varToWatch + 'Change', function (e) {
            respondToNewValue(e.detail);
        });
    }
    setTimeout(pollForChange, timeout);
}

Вызвав метод:

watchVariables(['username', 'userid']);

Он обнаружит изменения в имени пользователя и пользователя.

Ответ 16

Пожалуйста, ребята, помните, первоначальный вопрос был для ПЕРЕМЕННЫХ, а не для ОБЪЕКТОВ;)

в дополнение ко всем ответам выше, я создал крошечную библиотеку forTheWatch.js, которая использует тот же способ для отлова и обратного вызова для изменений в обычных глобальных переменных в javascript.

Совместим с переменными JQUERY, нет необходимости использовать OBJECTS, и вы можете при необходимости напрямую передавать массив из нескольких переменных.

Если это может быть полезно...: https://bitbucket.org/esabora/forthewatch
В основном вам просто нужно вызвать функцию:
watchIt("theVariableToWatch", "varChangedFunctionCallback");

И извините заранее, если не актуально.

Ответ 17

Опять же, я модно опаздываю на эту вечеринку. Я знаю, что это не чистый JavaScript или jQuery; однако, я работал с проектом в knockout.js и нашел, что метод subscribe() отлично подходит для запуска события (или уведомления), когда значение чего-то меняется. Вот простой пример:

var foo = ko.observable(false);

foo.subscribe(function () {
    console.log('`foo` has changed: ' + foo());
    if (foo() === true) {
        // do something here
    }
});

Конечно, subscribe() работает над нокаутом "наблюдаемые" не по регулярным переменным. Вот ссылка на страницу документации нокаута, охватывающая наблюдаемые и подписные подписки, для всех, кого это интересует: http://knockoutjs.com/documentation/observables.html

Ответ 18

Это старый поток, но я наткнулся на второй самый высокий ответ (пользовательские слушатели), ища решение с помощью Angular. Несмотря на то, что решение работает, угловое устройство имеет лучший встроенный способ решения этой проблемы с использованием @Output и событий. Отключение примера в ответе пользовательского прослушивателя:

ChildComponent.html

<button (click)="increment(1)">Increment</button>

ChildComponent.ts

import {EventEmitter, Output } from '@angular/core';

@Output() myEmitter: EventEmitter<number> = new EventEmitter<number>();

private myValue: number = 0;

public increment(n: number){
  this.myValue += n;

  // Send a change event to the emitter
  this.myEmitter.emit(this.myValue);
}

ParentComponent.html

<child-component (myEmitter)="monitorChanges($event)"></child-component>
<br/>
<label>{{n}}</label>

ParentComponent.ts

public n: number = 0;

public monitorChanges(n: number){
  this.n = n;
  console.log(n);
}

Теперь он будет обновлять n на родительском при каждом нажатии кнопки дочернего элемента. Рабочий блокblitz

Ответ 19

Utils = {
    eventRegister_globalVariable : function(variableName,handlers){
        eventRegister_JsonVariable(this,variableName,handlers);
    },
    eventRegister_jsonVariable : function(jsonObj,variableName,handlers){
        if(jsonObj.eventRegisteredVariable === undefined) {
            jsonObj.eventRegisteredVariable={};//this Object is used for trigger event in javascript variable value changes ku
        }
        Object.defineProperty(jsonObj, variableName , {
                    get: function() { 
                        return jsonObj.eventRegisteredVariable[variableName] },
                    set: function(value) {
                        jsonObj.eventRegisteredVariable[variableName] = value; handlers(jsonObj.eventRegisteredVariable[variableName]);}
                    });
            }