此篇文章是别处摘抄来的,因为道理浅显易懂
call 方法例子
JavaScript 中的 Array.prototype.slice.call(arguments)能将有 length 属性的对象转换为数组 ( 特别注意:这个对象一定要有 length 属性). 但有一个例外,IE 下的节点集合它不能转换(因为 IE 下的 dom 对象是以 com 对象的形式实现,js 对象和 com 对象不能进行转换)
首先,我们来看看 JavaScript 中的 slice 用法,在 JavaScript 中 Array 是一个类,slice 是此类中的一个方法,slice 的中文意思是‘截取’
一个是 String.slice => 返回值是字符串
一个是 Array.slice => 返回值是数组
Array.prototype.slice.call(arguments)能够将具有 length 属性的 arguments 转换为数组, 我们可以理解为就是 arguments.toArray().slice()
所以,这个过程我们是不是可以理解为 Array.prototype.slice.call(arguments)的实现过程就是把传入进来的具有 length 属性的第一个参数 arguments 转换为数组,再调用它的 slice(截取)方法
这个 Array.prototype.slice.call(arguments) 不单有 slice 方法,还有 call 方法。那么 call 方法又是如何用的呢
我们来看一个用 call 方法的例子:
; “ 复制代码 ”)
var a = function(){
console.log(this);
console.log(typeof this);
console.log(this instanceof String);
}
a.call(‘littleLuke’);
输出如下 // ‘littleLuke’ // Object // true
; “ 复制代码 ”)
以下是上面的代码片段在 Visual Studio Code 中的执行
从上面的代码片段中,我们可以看到调用了函数 a 的 call 方法之后,我们可以看到把 call 方法中的参数传入到函数 a 的作用域中去了,也就是说函数 a 中的 this 此时指向的就是它调用的 call 方法中的参数
我们上面说了,Array.Prototype.slice.call()除了 call 方法外,还有 slice 方法,那么 JavaScript 中 Array 的 slice 方法的内部实现是怎样的呢
; “ 复制代码 ”)
Array.prototype.slice = function(start,end){var result = new Array();
start \= start || 0;
end \= end || this.length; // this 指向调用的对象,当用了 call 后,能够改变 this 的指向,也就是指向传进来的对象,这是关键
for(var i = start; i < end; i++)
{result.push(this\[i\]);
} return result;
}
; “ 复制代码 ”)
所以整个过程我们基本就可以分 2 步进行理解了
Array.prototype.slice.call(arguments)
理解第一步: 其中,arguments 是一个具有 length 属性的对象, 通过 call 这个方法,把 arguments 指向了 Array.prototype.slice 方法的作用域,也就是说通过 call 方法,让 Array.prototype.slice 对 arguments 对象进行操作
理解第二步: Array.prototype.slice 就是对该对象使用 Array 类的 slice 方法。但是呢 arguments 它又不是个 Array 对象
typeof arguments === “Object” // 不是 ”Array”
所以我们没法用 arguments.slice()方法,这样是会报错的。所以这里,我们需要使用 Array.prototype.slice,它的作用就是第一步把 arguments 转换为一个 Array 对象,第二步对转换后的 Array 对象使用 slice 方法
理解了整个过程后,我们来看一些具体的例子,
下面是我自己在 Visual Studio Code 中执行的截图
在文章的结尾,来说两个在实际开发中比较常用到的通用方法
1. 将函数的实际参数转换成数组的方法(3 个方法)
方法一 : var args = Array.prototype.slice.call(arguments);
方法二: var args = [].slice.call(arguments);
方法三:
2. 转换成数组的通用函数
下面是关于 call 方法的官方解释
https://www.w3school.com.cn/JavaScript 函数 Call
使用 call() 方法,您可以编写能够在不同对象上使用的方法。
函数是对象方法
在 JavaScript 中,函数是对象的方法。
如果一个函数不是 JavaScript 对象的方法,那么它就是全局对象的函数
JavaScript call() 方法
call() 方法是预定义的 JavaScript 方法。
它可以用来调用所有者对象作为参数的方法。
通过 call(),您能够使用属于另一个对象的方法。
本例调用 person 的 fullName 方法,并用于 person1:
实例
var person = {fullName: function() {return this.firstName + " " + this.lastName;}
}
var person1 = {
firstName:"Bill",
lastName: "Gates",
}
var person2 = {
firstName:"Steve",
lastName: "Jobs",
}
person.fullName.call(person1); // 将返回 "Bill Gates"
Function.prototype.call()
mdn call
js 中 apply()和 call()的使用场景
猫狗实例
在 javascript OOP 中,我们经常会这样定义:
function cat() {}
cat.prototype = {
food: "fish",
say: function () {alert("I love" + this.food);
}
};
var blackCat = new cat;
//blackCat.say();
var whiteDog = {food: "bone"};
blackCat.say.call(whiteDog);
console.log(whiteDog);
但是如果我们有一个对象 whiteDog = {food:”bone”}, 我们不想对它重新定义 say 方法,那么我们可以通过 call 或 apply 用 blackCat 的 say 方法:blackCat.say.call(whiteDog);
可以看出 call 和 apply 是为了动态改变 this 而出现的,当一个 object 没有某个方法,但是其他的有,我们可以借助 call 或 apply 用其它对象的方法来操作。