听起来你想在这里做一些面向方面的编程......
JavaScript 作为一种 ECMAScript 方言,确实具有 可调用对象 的概念。每个可调用对象都有一个名为[[Call]] 的内部属性。该属性在 ECMA-262 规范第 5 版的第 8.6.2 节表 9 中进行了描述。它说:
执行与对象关联的代码。通过函数调用表达式调用。 SpecOp 的参数是一个 this 对象和一个包含传递给函数调用表达式的参数的列表。实现此内部方法的对象是可调用的。只有作为宿主对象的可调用对象才能返回引用值。
但需要注意的是 [[Call]] 是一个内部属性,规范中说:
内部属性没有名称,不能通过 ECMAScript 语言运算符直接访问。内部属性的存在纯粹是为了规范目的。
所以你不能在你自己的 JavaScript 代码中挂钩这个机制。
现在有两个方法在Function.prototype、apply 和call 中定义。确实,如果您更改Function.prototype.call 的定义,那么如果您创建自己的函数f,那么f.call 确实(除非在f 的原型或f 本身中被覆盖)执行该代码。如您所料,这不会通过直接调用f 自动发生。您必须显式调用call 方法。
话虽如此,最好不要在内置对象的原型中使用预定义的标准方法。库和应用程序中的许多现有代码都依赖于Function.prototype.call。不要惹它。当然,您可以通过多种方式实现一种 AOP 行为。一种是向Function.prototype 添加其他方法,但也不要这样做。另一种是使用 before 和 after 钩子编写自己的调用方法:
function call(theThis, before, main, after, beforeArgs, mainArgs, afterArgs) {
before.apply(theThis, beforeArgs);
main.apply(theThis, mainArgs);
after.apply(theThis. afterArgs);
}