Из другого ответа, который я недавно опубликовал, это в V8, и я думаю, что JavaScriptCore, но не Firefox, и это не спецификация. Поскольку вы можете перехватить операцию и компараторы, вы можете реализовать собственную перегрузку оператора в большинстве ситуаций, приложив немного усилий.
var actions = [];
var overload = {
valueOf: function(){
var caller = arguments.callee.caller;
actions.push({
operation: caller.name,
left: caller.arguments[0] === this ? "unknown" : this,
right: caller.arguments[0]
});
return Object.prototype.toString.call(this);
}
};
overload.toString = overload.valueOf;
overload == 10;
overload === 10;
overload * 10;
10 / overload;
overload in window;
-overload;
+overload;
overload < 5;
overload > 5;
[][overload];
overload == overload;
console.log(actions);
Выход:
[ { operation: 'EQUALS',
left: overload,
right: 10 },
{ operation: 'MUL',
left: overload,
right: 10 },
{ operation: 'DIV',
left: 'unknown',
right: overload },
{ operation: 'IN',
left: overload,
right: DOMWindow },
{ operation: 'UNARY_MINUS',
left: overload,
right: undefined },
{ operation: 'TO_NUMBER',
left: overload,
right: undefined },
{ operation: 'COMPARE',
left: overload,
right: 5 },
{ operation: 'COMPARE',
left: 'unknown',
right: overload },
{ operation: 'ToString',
left: 'unknown',
right: overload } ]
На данный момент у вас есть все входы и операция, поэтому оставшаяся часть является результатом операции. Получатель операции получит примитивное значение, строку или число, и вы не можете этого предотвратить. Если это не произвольный приемник, скажем, экземпляр класса, который вы перегрузили оператором, вы можете обрабатывать различные ловушки получения / установки для перехвата входящего значения / предотвращения перезаписи. Вы можете сохранить операнды и операцию в некотором центральном поиске и использовать простой метод для отслеживания примитивного значения до операции, которая его произвела, а затем создать любую логику, которую вы хотите выполнить для своей пользовательской операции. Другой метод, который позволит произвольным приемникам, которые впоследствии могут быть преобразованы в сложные формы, будет заключаться в кодировании данных в примитивное значение, чтобы его можно было вернуть обратно в ваш сложный класс. Например, значение RGB из 3 различных 8-битных целых чисел (255,255,255) может быть преобразовано в одно число на стороне получения, а сторона получателя может тривиально преобразовать его обратно в его сложные компоненты. Или для более сложных данных вы даже можете вернуть сериализованную строку JSON.
Наличие доступа к прокси-серверам Harmony (Firefox6 +, Nodejs с флагом) значительно упрощает весь этот процесс, поскольку вы можете создавать прокси-серверы захвата практически для всего, анализировать весь процесс от начала до конца и делать все, что вы хотите. Экземпляры операндов ваших данных / класса, valueOf / toString / геттеры всех возможных значений, к которым может обращаться внутренний механизм, любой объект-получатель, о котором вы заранее осведомлены, и даже перехват произвольных получателей в случаеwith(trappingProxy){ "all variable lookup, creation, and setting in here invokes traps on our proxy"; }