乐趣区

JavaScript中的Arrayprototypeslicecall方法

此篇文章是别处摘抄来的,因为道理浅显易懂

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 用其它对象的方法来操作。

退出移动版