直接说结论:this 的绑定方式本身并不直接左右方法寻址的效率,真正决定性能的关键在于方法调用时对象结构是否稳定、访问路径是否一致——而这正是内联缓存(IC)能否有效命中或失效的核心因素。V8 并不会专门为“this 是谁”做特殊优化,但它会为类似 obj.method() 的调用路径生成 IC stub;一旦 this 所指向的对象形状变得混乱,或访问模式频繁变化,IC 便会降级。此时,方法寻址会从“直接按偏移量读取”退化为“全量属性查找”,性能差距瞬间可达 3–5 倍。
点号调用 + 稳定 this → IC 高效命中
当 this 指向一个隐藏类(Hidden Class)始终一致的对象,并且始终通过 this.method 这种形式调用时,V8 能够在热路径上建立单态 IC:
- IC 会记录该对象的 Hidden Class,以及
method属性在内存中的固定偏移量。 - 后续调用完全无需遍历原型链,直接按偏移量跳转到函数地址,如同按图索骥一般。
- 典型场景是 class 实例的方法调用(例如
user.getName()),构造时所有属性均已声明,没有动态增删,形状稳定不变。
bind / call / apply → 多一层间接,IC 无法穿透
func.bind(obj) 或 func.call(obj) 创建的调用,本质上绕过了原始对象的属性访问路径:
boundFn()是一个新函数,其内部通过闭包获取原始func和绑定的this,根本不走obj.method那个点号路径。- V8 无法为
boundFn()建立针对obj的方法寻址 IC,因为调用目标并非obj上的属性,而是闭包中保存的函数引用。 - 每次执行都需要触发闭包环境查找及参数重组,TurboFan 也无法内联,堆内存占用更高。
箭头函数与 this 无关,但影响方法归属路径
箭头函数没有自己的 this,本身也不参与方法寻址逻辑——它根本不会出现在 obj.xxx() 的属性查找链中:
- 它并非作为对象属性存在,通常定义在词法作用域内(如 class 方法体中),属于闭包变量。
- 调用
handler()就是直接调用函数值,无需经过任何对象属性访问,IC 完全无法介入。 - 好处是没有 this 绑定开销,但缺点是——因为没有属性访问,IC 无法加速;并且它也不能复用对象上的方法共享机制。
隐式绑定断裂 → IC 瞬间失效
表面上只是 this 发生了变化,但本质是对象结构或访问模式被破坏:
- 例如
setTimeout(obj.method, 100):取值后obj.method变成了纯函数引用,调用时 this 为 undefined,但更关键的是——这次调用脱离了obj的 Hidden Class 上下文,IC 立即失效。 - 再如混用形状不同的对象:
obj1 = { method() {} }与obj2 = { method() {}, extra: true },即便都用objX.method()调用,IC 也会被迫升级为多态甚至超态,性能随之下降。 - 还有动态添加属性:
obj.method = () => {}之后再调用,Hidden Class 已经改变,IC 中缓存的旧信息作废,一切重新开始。
