返回
作为一个前端程序媛,在提升学习的道路上,不可避免的与 apply 和 call 相遇了。之前由于它俩出镜率有点低,都静静的擦肩而过了!今天不小心被它俩的魅力所吸引,加上本小姐心情好,就让我们好好的相识一下吧 O(∩_∩)O~
ECAMScript 3 给 Function 的原型定义了两个方法,它们是 Function.prototype.call 和 Function.prototype.apply。
Function.prototype.call 和 Function.prototype.apply 都是非常常用的方法。
apply 接受两个参数,第一个参数指定了函数体内 this 对象的指向, 第二个参数为一个带下标的集合,这个集合可以为数组,也可以为类数组,apply 方法把这个集合中的元素作为参数传递给被调用的函数
var func = function(a, b, c) {
console.log([a, b, c]); //输出:[1,2,3]
};
func.apply(null, [1, 2, 3]);
call 传入的参数数量不固定, 跟 apply 相同的是,第一个参数也是代表函数体内的 this 指向, 从第二个参数开始往后,每个参数被依次传入函数
var func = function(a, b, c) {
console.log([a, b, c]); //输出:[1,2,3]
};
func.call(null, 1, 2, 3);
var func = function(a, b, c) {
console.log(this === window); // 输出:true
};
func.apply(null, [1, 2, 3]);
但如果是在严格模式下,函数体内的 this 还是为 null
var func = function(a, b, c) {
"use strict";
console.log(this === null); // 输出:true
};
func.apply(null, [1, 2, 3]);
var a = Math.max.apply(null, [1, 2, 5, 3, 4]);
console.log(a); // 输出:5
例一
var obj1 = {
name: "sven",
};
var obj2 = {
name: "anne",
};
window.name = "window";
var getName = function() {
console.log(this.name);
};
getName(); // 输出: window
getName.call(obj1); // 输出: sven
getName.call(obj2); // 输出: anne
其中在执行
getName.call(obj1);
时,类似于执行
var getName = function() {
console.log(obj1.name); // 输出: sven
};
例二
document.getElementById("div1").onclick = function() {
console.log(this.id); // 输出: div1
var func = function() {
console.log(this.id); // 输出: undefined
};
func();
};
//修正后
document.getElementById("div1").onclick = function() {
var func = function() {
console.log(this.id); // 输出: div1
};
func.call(this);
};
Function.prototype.bind = function(context) {
var self = this; // 保存原函数
return function() {
// 返回一个新的函数
return self.apply(context, arguments); //执行新的函数的时候,会把之前传入的context当作新的函数体的this
};
};
var obj = {
name: "sven",
};
var func = function() {
console.log(this.name); // 输出: sven
}.bind(obj);
func();
//复杂化
Function.prototype.bind = function() {
var self = this, // 保存原函数
context = [].shift.call(arguments), //需要绑定的this上下文
args = [].slice.call(arguments); //剩余的参数转成数组
return function() {
// 返回一个新的函数
return self.apply(context, [].concat.call(args, [].slice.call(arguments)));
//执行新的函数的时候,会把之前传入的context当作新的函数体的this
//并且组合两次分别传入的参数,作为新的函数的参数
};
};
var obj = {
name: "sven",
};
var func = function(a, b, c, d) {
console.log(this.name); // 输出: sven
console.log([a, b, c, d]); //输出: [1,2,3,4]
}.bind(obj, 1, 2);
func(3, 4);
借用方法的第一种场景是“借用构造函数”,通过这种技术,可以实现一些类似继承的效果
var A = function(name) {
this.name = name;
};
var B = function() {
A.apply(this, arguments);
};
B.prototype.getName = function() {
return this.name;
};
var b = new B("sven");
console.log(b.getName()); // 输出: 'sven'
借用方法的第二种场景——函数的参数列表 arguments 是一个类数组对象,虽然它也有“下标”,但它并非真正的数组,所以也不能像数组一样,进行排序操作或者往集合里添加一个新的元素。
var a = {};
Array.prototype.push.call(a, "first");
console.log(a.length); // 输出: 1
console.log(a[0]); //输出: first
//这段代码在大部分浏览器里都能顺利执行,但由于引擎的内部实现存在差异,如果在低版本的 IE浏览器 中执行,必须显式地给对象 a 设置 length属性
var a = {
length: 0,
};
借用 Array.prototype.push 方法的对象还要满足以下两个条件