在看关于javascript继承的时候 好多地方都用到了apply()和call() 之前在简单编程的时候根本没有遇到过 查阅资料后才发现这两个方法是如此的好用。

下面从几方面来看一下这两个方法:

  1、定义

  2、从例子中详解

  3、在实际应用中让人意想不到的拓展

一、首先从定义入手:

  apply()和call()都是定义在Function对象原型上的方法(Function.prototype.apply = function(){}&Function.prototype.call = function(){}),故可以直接使用。

  apply():调用函数,并用指定对象替换函数的 this 值,同时用指定数组替换函数的参数。

    apply([thisObj[,argArray]])

    参数

      thisObj可选。 要用作 this 对象的对象。
      argArray可选。 要传递到函数的一组参数。
    备注
       如果 argArray 不是有效对象,则会出现“应为对象”错误。如果既未提供 argArray 也未提供 thisObj,则原始 this 对象将被用作 thisObj 且不会传递任何参数。
 
  call():调用一个对象的方法,用另一个对象替换当前对象。

     call([thisObj[, arg1[, arg2[, [, argN]]]]])
    参数
      thisObj可选。 将作为当前对象使用的对象。
      arg1, arg2, , argN可选。 将被传递到该方法的参数列表。
    备注

       call 方法用于调用代表另一项目的方法。 它允许您将函数的 this 对象从初始上下文变为由 thisObj 指定的新对象。如果没有提供 thisObj 参数,则 global 对象被用作 thisObj。
二、例子(理解的关键)
      1、apply()示例           
<script type="text/javascript">
function Animal(name,color){
this.name = name;
this.color = color;
this.type = "animal";
} function Bird(name,color,age){
Animal.apply(this,arguments);//此时Animal中this 指向的是windows故需要使用apply()方法将this的指向改为 Bird实例
this.age = age;
} var sparrow = new Bird("bage","yellow","1");
console.log("Hello,my name is "+sparrow.name+",my color is "+sparrow.color+".I am "+sparrow.age+" years old.I am a "+sparrow.type);      
</script> 

//输出结果:Hello,my name is bage,my color is yellow.I am 1 years old.I am a animal   

这个简单的例子实现了一个Bird继承Animal构造函数的方法,如果换为一下代码:  

<script type="text/javascript">
function Animal(name,color){
this.name = name;
this.color = color;
this.type = "animal";
} function Bird(name,color,age){
Animal(name,color);
this.age = age;
} var sparrow = new Bird("bage","yellow","1");
console.log("Hello,my name is "+sparrow.name+",my color is "+sparrow.color+".I am "+sparrow.age+" years old.I am a "+sparrow.type); </script>

//输出结果:Hello,my name is undefined,my color is undefined.I am 1 years old.I am a undefined ×

在apply()中的参数arguemts 是Javascript Function对象的属性(Function对象的7个属性和5个方法 https://msdn.microsoft.com/zh-cn/library/x844tc74(v=vs.94).aspx),获取当前正在执行的 Function 对象的参数, arguments 对象中包含的各个参数的访问方式与数组元素的访问方式相同。

  即,在本例中指的是["bage","yellow","1"],这也是apply与call最主要的区别。

而,apply()中的参数this 指向是Bird的实例sparrow

  2、call()  

  <script type="text/javascript>
function Animal(name,color){
this.name = name;
this.color = color;
this.type = "animal";
} function Bird(name,color,age){
Animal.call(this,name,color);
this.age = age;
} var sparrow = new Bird("bage","yellow","1");
console.log("Hello,my name is "+sparrow.name+",my color is "+sparrow.color+".I am "+sparrow.age+" years old.I am a "+sparrow.type); </script>

//输出结果:Hello,my name is bage,my color is yellow.I am 1 years old.I am a animal

3、什么时候用apply() 什么时候用call()

 <script type="text/javascript>
function Animal(color,name){
this.name = name;
this.color = color;
this.type = "animal";
} function Bird(name,color,age){
Animal.apply(this,arguments);
this.age = age;
} var sparrow = new Bird("bage","yellow","1");
console.log("Hello,my name is "+sparrow.name+",my color is "+sparrow.color+".I am "+sparrow.age+" years old.I am a "+sparrow.type); </script>

//输出结果:Hello,my name is yellow,my color is bage.I am 1 years old.I am a animal ×

<script type="text/javascript">
function Animal(color,name){
this.name = name;
this.color = color;
this.type = "animal";
} function Bird(name,color,age){
Animal.call(this,color,name);
this.age = age;
} var sparrow = new Bird("bage","yellow","1");
console.log("Hello,my name is "+sparrow.name+",my color is "+sparrow.color+".I am "+sparrow.age+" years old.I am a "+sparrow.type); </script>

//输出结果Hello,my name is bage,my color is yellow.I am 1 years old.I am a animal

从两个例子中不难看出,apply()的arguments 不会考虑“子类”和“父类”之间的参数顺序问题,默认参数顺序一致,当不一致时会出现赋值的混乱,而使用call()手动赋值可以保证与“父类”的参数一致。  

   故,能保证继承参数一致时使用apply(),无法确定时使用call().
 
三、在实际应用中意想不到的拓展

  细心的人可能已经察觉到,在我调用apply方法的时候,第一个参数是对象(this), 第二个参数是一个数组集合, 在调用Person的时候,他需要的不是一个数组,但是为什么他给我一个数组我仍然可以将数组解析为一个一个的参数,这个就是apply的一个巧妙的用处,可以将一个数组默认的转换为一个参数列表([param1,param2,param3] 转换为 param1,param2,param3) 这个如果让我们用程序来实现将数组的每一个项,来装换为参数的列表,可能都得费一会功夫,借助apply的这点特性,所以就有了以下高效率的方法:

  1、Math.max 可以实现得到数组中最大的一项

    因为Math.max 参数里面不支持Math.max([param1,param2]) 也就是数组

    但是它支持Math.max(param1,param2,param3…),所以可以根据刚才apply的那个特点来解决 var max=Math.max.apply(null,array),这样轻易的可以得到一个数组中最大的一项(apply会将一个数组装换为一个参数接一个参数的传递给方法)

    这块在调用的时候第一个参数给了一个null,这个是因为没有对象去调用这个方法,我只需要用这个方法帮我运算,得到返回的结果就行,.所以直接传递了一个null过去

  2、Math.min  可以实现得到数组中最小的一项

    同样和 max是一个思想 var min=Math.min.apply(null,array);

  3、Array.prototype.push 可以实现两个数组合并

    同样push方法没有提供push一个数组,但是它提供了push(param1,param,…paramN) 所以同样也可以通过apply来装换一下这个数组,即:

1      vararr1=new Array("1","2","3");
2
3 vararr2=new Array("4","5","6");
4
5 Array.prototype.push.apply(arr1,arr2);

    也可以这样理解,arr1调用了push方法,参数是通过apply将数组装换为参数列表的集合.

通常在什么情况下,可以使用apply类似Math.min等之类的特殊用法:

一般在目标函数只需要n个参数列表,而不接收一个数组的形式([param1[,param2[,…[,paramN]]]]),可以通过apply的方式巧妙地解决这个问题!

(注:第三部分来自网络http://www.cnblogs.com/KeenLeung/archive/2012/11/19/2778229.html)
 

不用不知道 apply()与call()的强大的更多相关文章

  1. 函数的属性和方法之call、apply 及bind

    一.前言 ECMAScript中的函数是对象,因此函数也有属性和方法.每个函数都包含两个属性:length和prototype.每个函数也包含两个非继承来的方法:apply()和call(),还有一些 ...

  2. 函数对象的apply()和call()方法

    每个函数都包含两个非继承而来的方法:apply()和call().这两个方法的用途都是在特定的作用域中调用函数,特定的作用域为this参数指定的对象. apply()和call()真正强大的地方是能够 ...

  3. 泛函编程(25)-泛函数据类型-Monad-Applicative

    上两期我们讨论了Monad.我们说Monad是个最有概括性(抽象性)的泛函数据类型,它可以覆盖绝大多数数据类型.任何数据类型只要能实现flatMap+unit这组Monad最基本组件函数就可以变成Mo ...

  4. javascript高级程序设计一(80-116)

    81.函数内部属性:arguments.arguments.callee.this. window.color = "red"; var o={color:"blue&q ...

  5. R语言︱机器学习模型评估方案(以随机森林算法为例)

    笔者寄语:本文中大多内容来自<数据挖掘之道>,本文为读书笔记.在刚刚接触机器学习的时候,觉得在监督学习之后,做一个混淆矩阵就已经足够,但是完整的机器学习解决方案并不会如此草率.需要完整的评 ...

  6. 快速学习 javascript

    // js 6种数据类型:字符串.数值.布尔值.undefined.null.object // 三种非数字的数字类型:Infinity -Infinity NaN var str = "H ...

  7. JavaScript读书笔记(6)-Function

    Function类型 ECMAScript中函数是对象,每个函数都是Function类型的实例,也有属性和方法,函数是对象,函数名实际上市一个指向函数对象的指针,不会与某个函数绑定: function ...

  8. 关于javascript中this 指向的4种调用模式

    this指向问题绝对可以排js 的top 5最难和最重点的问题,初学者常常搞不清楚this指向哪里,特别是学过java和c#的人,想当年俺也迷糊了好久,直到遇到蝴蝶书,主要是因为js和主流的面向对象语 ...

  9. javascript中apply和eval结合的强大用法

        eval是一个函数,可以接受一个参数,这个参数可以作为js语句被解释性的执行,利用这个特性,eval和apply结合起来,可以大大简化代码  如下例子 <a class="cl ...

随机推荐

  1. 01--从根源种子CCNode说起

    CCNode作为渲染框架的基类(暂且这样理解,CCObject为引擎基类)其中定义了绘制游戏元素相关的属性以及相关方法.属性当中需要注意的一个是Z坐标,在渲染框架中用来表示元素的遮挡关系,其值越小越容 ...

  2. php中的MVC模式运用

    [size=5][color=Red]php中的MVC模式运用[/color][/size] 首先我来举个例子: 一个简单的文章显示系统 简单期间,我们假定这个文章系统是只读的,也就是说这个例子将不涉 ...

  3. Hdu1076(n个闰年后的年份)

    #include <stdio.h> #include<stdlib.h> int main() { int T,Y,n,printYear; scanf("%d&q ...

  4. [POJ] 1064 Cable master (二分查找)

    题目地址:http://poj.org/problem?id=1064 有N条绳子,它们的长度分别为Ai,如果从它们中切割出K条长度相同的绳子,这K条绳子每条最长能有多长. 二分绳子长度,然后验证即可 ...

  5. 《Programming WPF》翻译 第9章 4.模板

    原文:<Programming WPF>翻译 第9章 4.模板 对一个自定义元素最后的设计考虑是,它是如何连接其可视化的.如果一个元素直接从FrameworkElement中派生,这将会适 ...

  6. InnoSetup XML操作函数

    用于InnoSetup 5 以上.对XML文件的操作,简化InnoSetup XML访问过程. 1. [代码]InnoSetup 5 脚本     { ======================== ...

  7. 【转】android 兼容性测试 CTS 测试过程(实践测试验证通过)

    原文网址:http://blog.csdn.net/jianguo_liao19840726/article/details/7222814 写这个博客的时候是为了记忆,建议大家还是看官方的说明,官方 ...

  8. SQL省市区三级表结构

    -- 表的结构 areaDROP TABLE  area;CREATE TABLE  area (   id int NOT NULL ,  areaID int NOT NULL,  area va ...

  9. Thinkphp 3.0版本上传文件加图片缩略图实例解析

    先看html加个表单,注意这里的action 路径要选 对. <div> <form action="__URL__/add_img" enctype=" ...

  10. GCD 延时操作

    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delayInSeconds * NSEC_PER_SEC)), dispatch_ ...