详解 JavaScript的 call() 和 apply()
程序员文章站
2022-06-16 23:18:30
定义 ECMAScript规范为所有函数都包含两个方法(这两个方法非继承而来), call 和 apply 。这两个函数都是在特定的作用域中调用函数,能改变函数的作用域,实际上是改变函数体内 this 的值 。 用法 我们看到通过方法本身的call 和 apply 执行了该函数。 我们改变了函数运行 ......
定义
ecmascript规范为所有函数都包含两个方法(这两个方法非继承而来), call
和 apply
。这两个函数都是在特定的作用域中调用函数,能改变函数的作用域,实际上是改变函数体内 this
的值 。
call 和 apply
语法 | 定义 | 说明 |
---|---|---|
call(thisobj,object) | 调用一个对象的一个方法,以另一个对象替换当前对象。 | call 方法可以用来代替另一个对象调用一个方法。call 方法可将一个函数的对象上下文从初始的上下文改变为由 thisobj 指定的新对象.如果没有提供 thisobj 参数,那么 global 对象被用作 thisobj |
apply(thisobj,[argarray]) | 应用某一对象的一个方法,用另一个对象替换当前对象。 | 如果 argarray 不是一个有效的数组或者不是 arguments 对象,那么将导致一个 typeerror。如果没有提供 argarray 和 thisobj 任何一个参数,那么 global 对象将被用作 thisobj, 并且无法被传递任何参数 |
用法
调用函数,传递参数
//定义一个add 方法 function add(x, y) { return x + y; } //用call 来调用 add 方法 function myaddcall(x, y) { //调用 add 方法 的 call 方法 return add.call(this, x, y); } //apply 来调用 add 方法 function myaddapply(x, y) { //调用 add 方法 的 applly 方法 return add.apply(this, [x, y]); } console.log(myaddcall(10, 20)); //输出结果30 console.log(myaddapply(20, 20)); //输出结果40
我们看到通过方法本身的call
和 apply
执行了该函数。
改变函数作用域
var name = '小白'; var obj = {name:'小红'}; function sayname() { return this.name; } console.log(sayname.call(this)); //输出小白 console.log(sayname. call(obj)); //输入小红
我们改变了函数运行的作用域, 通过绑定不同的对象,函数内部 this
也不同。最终输入结果才会这样。
高级用法,实现 js 继承
//父类 person function person() { this.sayname = function() { return this.name; } } //子类 chinese function chinese(name) { //借助 call 实现继承 person.call(this); this.name = name; this.ch = function() { alert('我是中国人'); } } //子类 america function america(name) { //借助 call 实现继承 person.call(this); this.name = name; this.am = function() { alert('我是美国人'); } } //测试 var chinese = new chinese('成龙'); //调用 父类方法 console.log(chinese.sayname()); //输出 成龙 var america = new america('america'); //调用 父类方法 console.log(america.sayname()); //输出 america
区别
- 参数不同, apply 传入的是一个参数数组,也就是将多个参数组合成一个参数数组, call 从第二个参数开始依次传入.
- apply 可以直接将当前函数的arguments对象作为apply的第二个参数传入
结束语
call
和 apply
最大的好处:方便我们解耦,对象不需要和方法有任何的耦合性,能使我们写出更好的面相对象程序。
大家如果看一些 js 框架底层的话会看到好多地方都有大量用到。
上一篇: tensorflow gpu使用说明
下一篇: 索引_举例:IO成本:全表扫描<走索引