apply 和call 的区别,apply实用小技巧
Js apply方法详解
我在一开始看到javascript的函数apply和call时,非常的模糊,看也看不懂,最近在网上看到一些文章对apply方法和call的一些示例,总算是看的有点眉目了,在这里我做如下笔记,希望和大家分享.. 如有什么不对的或者说法不明确的地方希望读者多多提一些意见,以便共同提高.. 主要我是要解决一下几个问题: 1.apply和call的区别在哪里 2.什么情况下用apply,什么情况下用call 3.apply的其他巧妙用法(一般在什么情况下可以使用apply) 我首先从网上查到关于apply和call的定义,然后用示例来解释这两个方法的意思和如何去用. apply:方法能劫持另外一个对象的方法,继承另外一个对象的属性. Function.apply(obj,args)方法能接收两个参数
obj:这个对象将代替Function类里this对象
args:这个是数组,它将作为参数传给Function(args-->arguments) call:和apply的意思一样,只不过是参数列表不一样. Function.call(obj,[param1[,param2[,…[,paramN]]]])
obj:这个对象将代替Function类里this对象
params:这个是一个参数列表 1.apply示例: <script type="text/javascript">
/*定义一个人类*/
function Person(name,age) {
this.name=name; this.age=age;
}
/*定义一个学生类*/
functionStudent(name,age,grade) {
Person.apply(this,arguments); this.grade=grade;
}
//创建一个学生类
var student=new Student("qian",21,"一年级");
//测试
alert("name:"+student.name+"\n"+"age:"+student.age+"\n"+"grade:"+student.grade);
//大家可以看到测试结果name:qian age:21 grade:一年级
//学生类里面我没有给name和age属性赋值啊,为什么又存在这两个属性的值呢,这个就是apply的神奇之处.
</script> 分析: Person.apply(this,arguments); this:在创建对象在这个时候代表的是student arguments:是一个数组,也就是[“qian”,”21”,”一年级”]; 也就是通俗一点讲就是:用student去执行Person这个类里面的内容,在Person这个类里面存在this.name等之类的语句,这样就将属性创建到了student对象里面 2.call示例 在Studen函数里面可以将apply中修改成如下: Person.call(this,name,age); 这样就ok了 3.什么情况下用apply,什么情况下用call 在给对象参数的情况下,如果参数的形式是数组的时候,比如apply示例里面传递了参数arguments,这个参数是数组类型,并且在调用Person的时候参数的列表是对应一致的(也就是Person和Student的参数列表前两位是一致的) 就可以采用 apply , 如果我的Person的参数列表是这样的(age,name),而Student的参数列表是(name,age,grade),这样就可以用call来实现了,也就是直接指定参数列表对应值的位置(Person.call(this,age,name,grade)); 4.apply的一些其他巧妙用法 细心的人可能已经察觉到,在我调用apply方法的时候,第一个参数是对象(this), 第二个参数是一个数组集合, 在调用Person的时候,他需要的不是一个数组,但是为什么他给我一个数组我仍然可以将数组解析为一个一个的参数, 这个就是apply的一个巧妙的用处,可以将一个数组默认的转换为一个参数列表([param1,param2,param3] 转换为 param1,param2,param3) 这个如果让我们用程序来实现将数组的每一个项,来装换为参数的列表,可能都得费一会功夫,借助apply的这点特性,所以就有了以下高效率的方法: a)Math.max 可以实现得到数组中最大的一项 因为Math.max 参数里面不支持Math.max([param1,param2]) 也就是数组 但是它支持Math.max(param1,param2,param3…),所以可以根据刚才apply的那个特点来解决 var max=Math.max.apply(null,array),这样轻易的可以得到一个数组中最大的一项 (apply会将一个数组装换为一个参数接一个参数的传递给方法) 这块在调用的时候第一个参数给了一个null,这个是因为没有对象去调用这个方法,我只需要用这个方法帮我运算,得到返回的结果就行,.所以直接传递了一个null过去 b)Math.min 可以实现得到数组中最小的一项 同样和 max是一个思想 var min=Math.min.apply(null,array); c)Array.prototype.push 可以实现两个数组合并 同样push方法没有提供push一个数组,但是它提供了push(param1,param,…paramN) 所以同样也可以通过apply来装换一下这个数组,即: vararr1=new Array("1","2","3");
vararr2=new Array("4","5","6");
Array.prototype.push.apply(arr1,arr2); 也可以这样理解,arr1调用了push方法,参数是通过apply将数组装换为参数列表的集合. 通常在什么情况下,可以使用apply类似Math.min等之类的特殊用法: 一般在目标函数只需要n个参数列表,而不接收一个数组的形式([param1[,param2[,…[,paramN]]]]),可以通过apply的方式巧妙地解决这个问题! 5.总结: 一开始我对apply 非常的不懂,最后多看了几遍,自己多敲了几遍代码,才明白了中间的道理,所以,不管做什么事情,只要自己肯动脑子,肯动手敲代码,这样一个技术就会掌握… 还有比如第四部分得内容,巧妙的解决了实实在在存在的问题,这个肯定不是一个初学者能想到的解决方案(这个也不是我自己想的),没有对编程有一定认识的不会想到这个的,还是一句话,多积累,多学习,提升自己的能力和对编程思想的理解能力才是最关键!
原文地址:http://blog.csdn.net/business122/article/details/8000676
<li class="alt"><span><span>Js apply方法详解 </span></span></li><li class=""><span>我在一开始看到javascript的函数apply和call时,非常的模糊,看也看不懂,最近在网上看到一些文章对apply方法和call的一些示例,总算是看的有点眉目了,在这里我做如下笔记,希望和大家分享.. 如有什么不对的或者说法不明确的地方希望读者多多提一些意见,以便共同提高.. </span></li><li class="alt"><span> </span></li><li class=""><span>主要我是要解决一下几个问题: </span></li><li class="alt"><span> </span></li><li class=""><span>1.apply和call的区别在哪里 </span></li><li class="alt"><span> </span></li><li class=""><span>2.什么情况下用apply,什么情况下用call </span></li><li class="alt"><span> </span></li><li class=""><span>3.apply的其他巧妙用法(一般在什么情况下可以使用apply) </span></li><li class="alt"><span> </span></li><li class=""><span>我首先从网上查到关于apply和call的定义,然后用示例来解释这两个方法的意思和如何去用. </span></li><li class="alt"><span> </span></li><li class=""><span> apply:方法能劫持另外一个对象的方法,继承另外一个对象的属性. </span></li><li class="alt"><span> </span></li><li class=""><span> Function.apply(obj,args)方法能接收两个参数 </span></li><li class="alt"><span>obj:这个对象将代替Function类里<span class="keyword">this</span><span>对象 </span></span></li><li class=""><span>args:这个是数组,它将作为参数传给Function(args-->arguments) </span></li><li class="alt"><span> </span></li><li class=""><span> call:和apply的意思一样,只不过是参数列表不一样. </span></li><li class="alt"><span> </span></li><li class=""><span> Function.call(obj,[param1[,param2[,…[,paramN]]]]) </span></li><li class="alt"><span>obj:这个对象将代替Function类里<span class="keyword">this</span><span>对象 </span></span></li><li class=""><span>params:这个是一个参数列表 </span></li><li class="alt"><span> </span></li><li class=""><span>1.apply示例: </span></li><li class="alt"><span> </span></li><li class=""><span><script type=<span class="string">"text/javascript"</span><span>> </span></span></li><li class="alt"><span><span class="comment">/*定义一个人类*/</span><span> </span></span></li><li class=""><span><span class="keyword">function</span><span> Person(name,age) { </span></span></li><li class="alt"><span> <span class="keyword">this</span><span>.name=name; </span><span class="keyword">this</span><span>.age=age; </span></span></li><li class=""><span>} </span></li><li class="alt"><span> <span class="comment">/*定义一个学生类*/</span><span> </span></span></li><li class=""><span>functionStudent(name,age,grade) { </span></li><li class="alt"><span> Person.apply(<span class="keyword">this</span><span>,arguments); </span><span class="keyword">this</span><span>.grade=grade; </span></span></li><li class=""><span>} </span></li><li class="alt"><span><span class="comment">//创建一个学生类 </span><span> </span></span></li><li class=""><span><span class="keyword">var</span><span> student=</span><span class="keyword">new</span><span> Student(</span><span class="string">"qian"</span><span>,21,</span><span class="string">"一年级"</span><span>); </span></span></li><li class="alt"><span><span class="comment">//测试 </span><span> </span></span></li><li class=""><span>alert(<span class="string">"name:"</span><span>+student.name+</span><span class="string">"\n"</span><span>+</span><span class="string">"age:"</span><span>+student.age+</span><span class="string">"\n"</span><span>+</span><span class="string">"grade:"</span><span>+student.grade); </span></span></li><li class="alt"><span><span class="comment">//大家可以看到测试结果name:qian age:21 grade:一年级 </span><span> </span></span></li><li class=""><span><span class="comment">//学生类里面我没有给name和age属性赋值啊,为什么又存在这两个属性的值呢,这个就是apply的神奇之处. </span><span> </span></span></li><li class="alt"><span></script> </span></li><li class=""><span> </span></li><li class="alt"><span>分析: Person.apply(<span class="keyword">this</span><span>,arguments); </span></span></li><li class=""><span> </span></li><li class="alt"><span><span class="keyword">this</span><span>:在创建对象在这个时候代表的是student </span></span></li><li class=""><span> </span></li><li class="alt"><span>arguments:是一个数组,也就是[“qian”,”21”,”一年级”]; </span></li><li class=""><span> </span></li><li class="alt"><span>也就是通俗一点讲就是:用student去执行Person这个类里面的内容,在Person这个类里面存在<span class="keyword">this</span><span>.name等之类的语句,这样就将属性创建到了student对象里面 </span></span></li><li class=""><span> </span></li><li class="alt"><span> </span></li><li class=""><span> </span></li><li class="alt"><span>2.call示例 </span></li><li class=""><span> </span></li><li class="alt"><span>在Studen函数里面可以将apply中修改成如下: </span></li><li class=""><span> </span></li><li class="alt"><span>Person.call(<span class="keyword">this</span><span>,name,age); </span></span></li><li class=""><span> </span></li><li class="alt"><span>这样就ok了 </span></li><li class=""><span> </span></li><li class="alt"><span>3.什么情况下用apply,什么情况下用call </span></li><li class=""><span> </span></li><li class="alt"><span>在给对象参数的情况下,如果参数的形式是数组的时候,比如apply示例里面传递了参数arguments,这个参数是数组类型,并且在调用Person的时候参数的列表是对应一致的(也就是Person和Student的参数列表前两位是一致的) 就可以采用 apply , 如果我的Person的参数列表是这样的(age,name),而Student的参数列表是(name,age,grade),这样就可以用call来实现了,也就是直接指定参数列表对应值的位置(Person.call(<span class="keyword">this</span><span>,age,name,grade)); </span></span></li><li class=""><span> </span></li><li class="alt"><span>4.apply的一些其他巧妙用法 </span></li><li class=""><span> </span></li><li class="alt"><span>细心的人可能已经察觉到,在我调用apply方法的时候,第一个参数是对象(<span class="keyword">this</span><span>), 第二个参数是一个数组集合, </span></span></li><li class=""><span> </span></li><li class="alt"><span>在调用Person的时候,他需要的不是一个数组,但是为什么他给我一个数组我仍然可以将数组解析为一个一个的参数, </span></li><li class=""><span> </span></li><li class="alt"><span>这个就是apply的一个巧妙的用处,可以将一个数组默认的转换为一个参数列表([param1,param2,param3] 转换为 param1,param2,param3) 这个如果让我们用程序来实现将数组的每一个项,来装换为参数的列表,可能都得费一会功夫,借助apply的这点特性,所以就有了以下高效率的方法: </span></li><li class=""><span> </span></li><li class="alt"><span> </span></li><li class=""><span> </span></li><li class="alt"><span>a)Math.max 可以实现得到数组中最大的一项 </span></li><li class=""><span> </span></li><li class="alt"><span>因为Math.max 参数里面不支持Math.max([param1,param2]) 也就是数组 </span></li><li class=""><span> </span></li><li class="alt"><span>但是它支持Math.max(param1,param2,param3…),所以可以根据刚才apply的那个特点来解决 <span class="keyword">var</span><span> max=Math.max.apply(</span><span class="keyword">null</span><span>,array),这样轻易的可以得到一个数组中最大的一项 </span></span></li><li class=""><span> </span></li><li class="alt"><span>(apply会将一个数组装换为一个参数接一个参数的传递给方法) </span></li><li class=""><span> </span></li><li class="alt"><span> 这块在调用的时候第一个参数给了一个<span class="keyword">null</span><span>,这个是因为没有对象去调用这个方法,我只需要用这个方法帮我运算,得到返回的结果就行,.所以直接传递了一个</span><span class="keyword">null</span><span>过去 </span></span></li><li class=""><span> </span></li><li class="alt"><span>b)Math.min 可以实现得到数组中最小的一项 </span></li><li class=""><span> </span></li><li class="alt"><span>同样和 max是一个思想 <span class="keyword">var</span><span> min=Math.min.apply(</span><span class="keyword">null</span><span>,array); </span></span></li><li class=""><span> </span></li><li class="alt"><span>c)Array.prototype.push 可以实现两个数组合并 </span></li><li class=""><span> </span></li><li class="alt"><span>同样push方法没有提供push一个数组,但是它提供了push(param1,param,…paramN) 所以同样也可以通过apply来装换一下这个数组,即: </span></li><li class=""><span> </span></li><li class="alt"><span>vararr1=<span class="keyword">new</span><span> Array(</span><span class="string">"1"</span><span>,</span><span class="string">"2"</span><span>,</span><span class="string">"3"</span><span>); </span></span></li><li class=""><span> vararr2=<span class="keyword">new</span><span> Array(</span><span class="string">"4"</span><span>,</span><span class="string">"5"</span><span>,</span><span class="string">"6"</span><span>); </span></span></li><li class="alt"><span>Array.prototype.push.apply(arr1,arr2); </span></li><li class=""><span> </span></li><li class="alt"><span>也可以这样理解,arr1调用了push方法,参数是通过apply将数组装换为参数列表的集合. </span></li><li class=""><span> </span></li><li class="alt"><span>通常在什么情况下,可以使用apply类似Math.min等之类的特殊用法: </span></li><li class=""><span> </span></li><li class="alt"><span>一般在目标函数只需要n个参数列表,而不接收一个数组的形式([param1[,param2[,…[,paramN]]]]),可以通过apply的方式巧妙地解决这个问题! </span></li><li class=""><span> </span></li><li class="alt"><span> </span></li><li class=""><span>5.总结: </span></li><li class="alt"><span> </span></li><li class=""><span>一开始我对apply 非常的不懂,最后多看了几遍,自己多敲了几遍代码,才明白了中间的道理,所以,不管做什么事情,只要自己肯动脑子,肯动手敲代码,这样一个技术就会掌握… </span></li><li class="alt"><span> </span></li><li class=""><span>还有比如第四部分得内容,巧妙的解决了实实在在存在的问题,这个肯定不是一个初学者能想到的解决方案(这个也不是我自己想的),没有对编程有一定认识的不会想到这个的,还是一句话,多积累,多学习,提升自己的能力和对编程思想的理解能力才是最关键! </span></li>
apply 和call 的区别,apply实用小技巧的更多相关文章
- PHP 常用函数库和一些实用小技巧
PHP 常用函数库和一些实用小技巧 作者: 字体:[增加 减小] 类型:转载 包括文件读取函式,文件写入函式,静态页面生成函式,目录删除函式等 文件读取函式 //文件读取函式 function ...
- Vim实用小技巧
Vim实用小技巧 一些网络上质量较高的Vim资料 从我07年接触Vim以来,已经过去了8个年头,期间看过很多的Vim文章,我自己觉得非常不错,而且创作时间也比较近的文章有如下这些. Vim入门 目前为 ...
- svn checkout 实用小技巧
svn checkout 实用小技巧 by:授客 QQ:1033553122 问题描述: 用svn小乌龟软件,进行update,commit之前,先要把svn工作目录checkout到本地,那么问 ...
- 实用小技巧(一):UIScrollView中上下左右滚动方向的判断
https://www.jianshu.com/p/93e8459b6dae 2017.06.01 01:13* 字数 674 阅读 1201评论 0喜欢 1 2017.06.01 01:13* 字数 ...
- VC6.0实用小技巧
VC6.0的若干实用小技巧 .检测程序中的括号是否匹配 把光标移动到需要检测的括号(如大括号{}.方括号[].圆括号()和尖括号<>)前面,键入快捷键 “Ctrl+]”.如果括号匹配正确, ...
- 必看!macOS进阶不得不知的实用小技巧
不知道大家对使用苹果电脑的体验如何?您充分利用您的mac了吗?其实macOS上存在着许多快捷方式和技巧可以帮助简化我们的工作流程,提高效率,但是在日常生活中经常被人们忽略或者遗忘.以下是macdown ...
- 实用小技巧:Notepad++直接连接Linux
实用小技巧:Notepad++直接连接Linux 前言 号称编辑器之神的Vim对于只会用几个基础操作的本人而言,在编辑一些大型文本有那么些力不从心: 平时都是通过Xftp拖到本地,修改完后再覆盖回去: ...
- Maven在Eclipse中的实用小技巧
前言 我们在开发的工程中很多都是Maven项目,这样更加便于我们jar包的管理.而我们一般使用的IDE都是Eclipse,由于我们在日常的开发过程中会经常要用到一些Maven的操作,所以我今天 ...
- Visual Studio实用小技巧
有一个有关微软Office的笑话,说的是它的特性太多: 当你觉得自己发现了一个Office的新特性时,它已经存在很多年了. 本文将介绍一些在Visual Studio(免费下载)中很实用却被忽略的小技 ...
随机推荐
- java 面试基础总结(二)---多线程
1.实现多线程的三种方法 1.继成Thread 类,覆盖run()方法即可 2.implements Runnable接口 3.implements Callale接口,执行时通过FutureTask ...
- 3张表实现RBAC
管理员表 CREATE TABLE cqh_admin ( id smallint unsigned not null auto_increment comment 'id', username va ...
- Vuejs的指令及组件用法总结
vuejs介绍 Vue.js是当下很火的一个JavaScript MVVM库,它是以数据驱动和组件化的思想构建的.相比于Angular.js,Vue.js提供了更加简洁.更易于理解的API,使得我们能 ...
- Angular2入门:TypeScript的函数 - 剩余参数和箭头函数
- #2 安装Python
上一篇文章主要记录 了Python简介,相信你已经爱上了小P,俗话说的好:公欲善其事,必先利其器,所以本文将带领你安装Python3! Windows平台 1.确认Windows位数: 鼠标右击此电脑 ...
- [CF1082E] Increasing Frequency
Description 给定一个长度为 \(n\) 的数列 \(a\) ,你可以任意选择一个区间 \([l,r]\) ,并给区间每个数加上一个整数 \(k\) ,求这样一次操作之后数列中最多有多少个数 ...
- webpack打包去除map文件及其他一些配置
一.vue-cli(3.x)搭建的项目,webpack(3.x)打包时,生成的map文件很大,目前又不知道是干嘛用的,所以就直接去掉了. 方法: 修改sourceMap配置成为false. 1:在bu ...
- [android] 手机卫士绑定sim卡
更新: 收不到启动广播,查看知乎,好像是说高版本的系统都禁止了 还可以通过adb发送开机广播 adb shell am broadcast -a android.intent.action.BOOT_ ...
- Why is one loop so much slower than two loops?
Question: Suppose a1, b1, c1, and d1 point to heap memory and my numerical code has the following co ...
- 实现Java Socket 客户端服务端交互实例
SocketService.java package socket; import java.io.BufferedReader; import java.io.IOException; import ...