闲聊js中的apply、call和arguments
JavaScript提供了apply和call两种调用方式来确定函数中的this的指向,在现实编码中,我确实
很少接触到这两个方法。但很无奈,很多面试题都要考这两种方法,我又没怎么用到,所以我们先来
闲聊下他们到底有什么用和到底怎么用。
我们先来聊一下apply的用法吧,它是用来改变函数的指向的,说白了,就是指向了别的函数的作用域。
例如看一下下面这个例子。
var A={
name:"我是小A",
fun:function(){
console.log("大家好! "+this.name)
}
}
var B = {
name:"我是小B"
}
A.fun(); //大家好! 我是小A
A.fun.apply(B); //大家好! 我是小B
可以看出来,当我们使用了apply将引用指向了B时,A调用fun时并不是调用本身中的name 而是调用了B中的那么;
但如果我们在调用函数中并没用引用到this,那么是不是说apply就失去了意义?再来看一下下面的例子
var A={
name:"我是小A",
fun:function(num,money){
console.log("大家好! "+this.name+" 我有"+num+"张"+money+"块")
}
}
var B = {
name:"我是小B"
}
var monies = [10,20];
A.fun.apply(B,monies); //大家好! 我是小B 我有10张20块
可以看出:apply中
第一个参数为thisObject,调用时采用传入的thisObject代替函数体中this的指向
第二个参数传入一个数组,函数会用数组的值取代“参数列表"。
这里聊一下monies,就要说到参数列表,当然我们会想到arguments,在这里插入聊一下arguments。网
上arguments是具有数组某些特性的‘类数组‘(伪数组);其实JS并没有重载函数的功能,但是auguments对象
能够模拟重载。每个函数都有一个Arguments对象实例arguments,它引用着函数的实参,可以用数组下
标的方式”[]“引用arguments的元素。arguments.length为函数实参个数。
function test() {
var s = "";
for (var i = 0; i < arguments.length; i++) {
alert(arguments[i]);
s += arguments[i] + ",";
}
return s;
}
test("name", "age")
输出
name,age
在匿名函数中可以用arguements.callee引用函数自身。
例如 function(){
if(n>0) return 0;
else return n+arguments.callee(n+1);
}
回归正题,我们可以用arguments来代替函数中的参数
var A={
name:"我是小A",
fun:function(num,money){
console.log("大家好! "+this.name+" 我有"+arguments[0]+"张"+arguments[1]+"块")
}
}
var B = {
name:"我是小B"
}
var monies = [10,20];
A.fun.apply(B,monies); //大家好! 我是小B 我有10张20块
这里又有一个问题,可不可以用Array.prototype.shift.apply(arguments)来代替arguments[0]呢?直接看代码
var A={
name:"我是小A",
fun:function(num,money){
console.log("大家好! "+this.name+" 我有"+Array.prototype.shift.apply(arguments)+"张"+arguments[1]+"块")
}
}
var B = {
name:"我是小B"
}
var monies = [10,20];
A.fun.apply(B,monies); //大家好! 我是小B 我有10张undefined块
当我们使用Array.prototype.shift.apply(arguments)调用之后,arguments[0]就会被抹去,所以arguments[1]就会冒到arguments[0]上。
这可以看出arguments这个”伪数组“,除了不是”原型继承自’Array.prototype‘“职位,其他特征和数组是一样的。
最后,我们考虑一下,如果使用apply方法时,传入的第一个参数是null时,调用函数时,会发生什么情况,会不会报错呢!
不多说,直接上例子
var A={
name:"我是小A",
fun:function(num,money){
console.log("大家好! "+this.name+" 我有"+num+"张"+money+"块")
}
}
var B = {
name:"我是小B"
}
var monies = [10,20];
name="我是小C"
A.fun.apply(null,monies); //大家好! 我是小C 我有10张20块
可以看到 如果第一传入的参数是null的话,在函数提内的this会指向全局对象,在浏览器中就是window。
所以可以总结出两点:
1.如果函数中有明确使用this,那么this就会指向传入的第一个参数的作用域。
2.如果传入的第一参数为null时,this就会指向全局的作用域。
apply的另一种用法就是用于将数组分割为一个个元素。
例如想在数组中a[1,2,3,4]中寻找出最大的袁术出来。
如果直接调用Math.max(a);就会输出NaN,所以这时候我们可以这样
Math.max.apply(null,a); //输出4
聊完了apply之后,我们再聊一下call就更简单了,其实他们的作用都是一样的。他们唯一的区别就是
apply和call在形式参数上的不同,apply只能传入两个参数且,第二个参数传入的是数组,而call在
第二参数开始可以接受任意个参数。大家看下面的例子就一目了然了
var A={
name:"我是小A",
fun:function(num,money){
console.log("大家好! "+this.name+" 我有"+num+"张"+money+"块")
}
}
var B = {
name:"我是小B"
}
var monies = [10,20];
A.fun(monies[0],monies[1]); //大家好! 我是小A 我有10张20块
A.fun.apply(B,monies); //大家好! 我是小B 我有10张20块
A.fun.call(B,monies); //大家好! 我是小B 我有10,20张undefined块
A.fun.call(B,monies[1],monies[2]); //大家好! 我是小B 我有20张undefined块
闲聊js中的apply、call和arguments的更多相关文章
- 兄台息怒,关于arguments,您的想法和大神是一样一样的----闲聊JS中的apply和call
JavaScript提供了apply和call两种调用方式来确定函数体中this的指向,表现出来的特征就是:对象可以'借用'其他对象的方法.之前的几篇博客回顾了一些Web控件的一些开发方法,我们聊了如 ...
- 深入理解js中的apply、call、bind
概述 js中的apply,call都是为了改变某个函数运行时的上下文环境而存在的,即改变函数内部的this指向. apply() apply 方法传入两个参数:一个是作为函数上下文的对象,另外一个是作 ...
- js中的apply和call API
借用网上的一个例子: fun.call(this,arg1,arg2,arg3) fun.apply(this,arguments) this.fun(arg1,arg2,arg3) 三种方法等效. ...
- js中call apply方法的使用介绍
js call call 方法 请参阅 应用于:Function 对象 要求 版本 5.5 调用一个对象的一个方法,以另一个对象替换当前对象. call([thisObj[,arg1[, arg2[, ...
- js 中call,apply,bind的区别
call.apply.bind方法的共同点与区别: apply.call.bind 三者都是用来改变函数的this对象的指向: apply.call.bind 三者都可以利用后续参数传参: bind ...
- js中call(),apply(),以及prototype的含义
最近段时间主要学习前端去了,然而所遇到的一些问题我觉得有必要去深究一下 prototype: 1 js中有三种表达方法 类方法,属性方法,原型方法 function People(name) { th ...
- Js中的apply和call
1.call和apply都是为了改变某个函数运行时的上下文而存在的 2.也就是改变函数体内this的指向. 3.二者的作用完全一样,只是接受参数的方式不太一样. 4.call 需要把参数按顺序传递进去 ...
- js中的apply调用
今天看了阮一锋老师的一篇文章,感觉很明了对闭包的理解,尤其是文章中的apply的介绍 apply()是函数对象的一个方法,它的作用是改变函数的调用对象,它的第一个参数就表示改变后的调用这个函数的对象. ...
- js中函数的 this、arguments 、caller,call(),apply(),bind()
在函数内部有两个特殊的对象,arguments 和 this,还有一个函数对象的属性caller. arguments对象 arguments是一个类似数组的对象,包含着传入函数的所有参数. func ...
随机推荐
- 【转】Javascript全局变量var与不var的区别
相信你对全局变量一定不陌生,在函数作用域里用a=1这种形式定义的变量会是一个全局变量,在全局作用域里,用下面3种形式都可以创建对全局可见的命名: <script> var a = 1; b ...
- SQL Server 表的管理_关于表的操作增删查改的操作的详解(案例代码)
SQL Server 表的管理_关于表的操作增删查改的操作的详解(案例代码) 概述: 表由行和列组成,每个表都必须有个表名. SQL CREATE TABLE 语法 CREATE TABLE tabl ...
- C#之Redis为所欲为
一 Redis是一种支持多种数据结构的键值对数据库 1.1Redis下载地址 :https://github.com/MicrosoftArchive/Redis 建议下载 .msi结尾的应用程序进行 ...
- Hibernate Session总结
现在我们可以在 IDEA 下新建一个 Hibernate 项目,接着上次内容这次主要总结一下 Hibernate 的 Session,及其核心方法. Session 概述 Session 接口是 Hi ...
- ThreadPoolExecutor 学习笔记
线程池的奥义 在开发程序的过程中,很多时候我们会遇到遇到批量执行任务的场景,当各个具体任务之间互相独立并不依赖其他任务的时候,我们会考虑使用并发的方式,将各个任务分散到不同的线程中进行执行来提高任务的 ...
- 读《图解HTTP》有感-(返回结果的HTTP状态码)
写在前面 HTTP状态码是由服务端产生,用于告诉客户端,服务端处理结果的编码 正文 1.状态码的作用是什么?具有什么特征? 状态码的作用是当客户端向服务器发送请求时,描述服务器的响应结果(如:服务器正 ...
- String的charAt(int index)可用于遍历一个字符串的所有字符
charAt(int index)遍历一个字符串的所有字符实例 String name = "Whatisjava?"; for (int i = 0; i < name.l ...
- ZooKeeper的使用---命令端
一.进入命令行 ./bin/zkCli.sh 二.常用命令 命令 作用 范例 备注 connect host:port 连接其他zookeeper客户端 connect hadoop2:21 ...
- Linux时间子系统之(一):时间的基本概念
专题文档汇总目录 Notes:Linux时间基准点:Linux时间和broken-down time(struct tm):不同精度的时间表示time_t.timeval.timespec. 原文地址 ...
- vue项目如何刷新当前页面
1.场景 在处理列表时,常常有删除一条数据或者新增数据之后需要重新刷新当前页面的需求. 2.遇到的问题 1. 用vue-router重新路由到当前页面,页面是不进行刷新的 2.采用window.rel ...