当你看代码时,经常会看到以下情形:(在这个博客里面,参数context是执行上下文的意思,params是参数的意思)

Object.prototype.toString.call(context, params);

fn.call(context, params);

Array.prototype.slice.call(context, params);

等等。

不要惊奇,不要慌张,看完这篇文章下次你就不怕了,甚至自己也会经常这么写了。

先说以下call,apply先不提了,等你把call学会了,到文章末尾也就把apply学会了。

用途:改变this指向

Object.prototype.toString.call(context, params);

这句话就相当于执行toString函数,如:toString(params); 但是为什么不直接这样写呢?

先看一段代码的执行结果:

  1. var str = "stringstr";
  2. var num = 12;
  3. var arr = [1, 2];
  4. var obj = {name: "solid"};
  5.  
  6. console.log(str.toString()); // stringstr
  7. console.log(num.toString()); //
  8. console.log(arr.toString()); // 1,2
  9. console.log(obj.toString()); // [object Object]
  10.  
  11. console.log(Object.prototype.toString.call(str)); // [object String]
  12. console.log(Object.prototype.toString.call(num)); // [object Number]
  13. console.log(Object.prototype.toString.call(arr)); // [object Array]
  14. console.log(Object.prototype.toString.call(obj)); // [object Object]

如果你调用Object.prototype.toString()函数是不是可以查看变量的类型啊,进而进行别的逻辑。

在js中,基本上所有的变量都是从Object继承来的,但是他们又都改写了toString方法,所以调用自身的toString方法会得到重写后的函数的结果。有的并没有什么实际的用处,如字符串的toString方法。

下面看点实用的引用吧:

  1.  

// A程序员写了一个函数
function Person(name, age, sex){
this.name = name;
this.age = age;
this.sex = sex;
}

  1.  

// 现在一个对象里,你的需求里也需要这三个参数
var zhangsan = {};
// 你就可以这样借用他的函数为你效力
Person.call(zhangsan, "张三", 18, "男");
console.log(zhangsan); // {name: "张三", age: 18, sex: "男"}

  1. // 你可能还需要其他的信息,再写就可以了
  2. function say(){
  3. this.say = function (){
  4. console.log(this.name + "具有说话的能力");
  5. }
  6. }
  7. say.call(zhangsan);
  8. zhangsan.say(); // 张三具有说话的能力

在上面这个例子中,zhangsan这个对象借用了其他的函数实现了自己的功能。这样调用函数,是不是相当于把Person函数中的this指向了zhangsan,把say函数中的this指向了zhangsan?结果实现了调用其他构造函数的方法,为自己实现了功能。

开始学习的时候,我看call也不顺眼,就直接看成前面那个方法的执行。

当不需要改变this时,可以这样调

fn.call(null, params);

不过这个好像没有什么意义,还不如直接写fn(params);即调用函数。

使用方法:fn.call(context, params);

还有其他的如:

Object.prototype.toString.call(context, params);

Array.prototype.slice.call(context, params);

在这里相当于: toString.call(context, params);   slice.call(context, params);

只是toString方法和slice不是全局的,而是Object和Array原型上的一个方法,所以需要这样来写。

看到这里对你的理解有没有一点帮助呢?有没有感觉使用起来也很简单呢。

下面说说apply。

apply的用途和call一模一样,只是参数有区别。call的参数是用逗号隔开的,apply的参数是一个数组。例子具体见下面的代码

  1. Person.call(zhangsan, "张三", 18, "男");
  2. Person.apply(zhangsan, ["张三", 18, "男"]);
  3. // 这两句是等价的

apply什么时候用呢?看下面的例子把

  1. // 已知一个数组,求数组中的最大值
  2. var numArr = [12, 21, 9, 18, 100, 0];
  3. var max = Math.max.apply(null, numArr);
  4. console.log(max); // 100
  5.  
  6. // 下面这个求最大值的函数你可以直接收藏使用了。

function max(){
  return Math.max.apply(null, arguments);
}
console.log(max(12, 21, 9, 18, 100, 0)); // 100

  1.  

call和apply用途与使用方法的更多相关文章

  1. apply()和call()的方法

    apply()和call()的方法的区别 参考文档https://www.cnblogs.com/lengyuehuahun/p/5643625.html 一直都没太明白apply()与call()的 ...

  2. JavaScript中的apply()方法和call()方法使用介绍

    1.每个函数都包含两个非继承而来的方法:apply()和call(). 2.他们的用途相同,都是在特定的作用域中调用函数. 3.接收参数方面不同,apply()接收两个参数,一个是函数运行的作用域(t ...

  3. JavaScript的apply()方法和call()方法

    1 <script type="text/javascript"> 2 /*定义一个人类*/ 3 function Person(name,age) 4 { 5 thi ...

  4. 利用apply和arguments复用方法

    首先,有个单例对象,它上面挂了很多静态工具方法.其中有一个是each,用来遍历数组或对象. var nativeForEach = [].forEach var nativeMap = [].map ...

  5. 彻底理解了call()方法,apply()方法和bind()方法

    javascript中的每一个作用域中都有一个this对象,它代表的是调用函数的对象.在全局作用域中,this代表的是全局对象(在web浏览器中指的是window).如果包含this的函数是一个对象的 ...

  6. apply方法和call方法。函数属性与方法。

    每个函数都有length属性哥prototype属性. length属性表示的是函数接入参数的个数 在es引用类型语言中,prototype是保存它们所有实例方法的真正所在.换句话来说,类似于toSt ...

  7. AngularJs $scope 里面的$apply 方法和$watch方法

    Angular $scope 里面的$apply 方法 Scope提供$apply方法传播Model变化 <!DOCTYPE html> <html> <head> ...

  8. 【JavaScript】call和apply区别及使用方法

    一.方法的定义call方法: 语法:fun.call(thisArg[, arg1[, arg2[, ...]]])定义:调用一个对象的一个方法,以另一个对象替换当前对象.说明:call 方法可以用来 ...

  9. apply方法和call方法的详解2

    1.apply和call的区别在哪里 2.什么情况下用apply,什么情况下用call 3.apply的其他巧妙用法(一般在什么情况下可以使用apply) 我首先从网上查到关于apply和call的定 ...

随机推荐

  1. memcache 拓展

    需要安装Libevent.memcached.memcache. 参考网址:https://www.cnblogs.com/hejun695/p/5369610.html 启动:/usr/local/ ...

  2. [转] Blob对象

    Blob是计算机界通用术语之一,全称写作:BLOB(binary large object),表示二进制大对象.MySql/Oracle数据库中,就有一种Blob类型,专门存放二进制数据.在javas ...

  3. java list转换json格式

    /** * 处理返回值(转换json格式和补零) * * @param resultDto5List * @param dateList * @return */ private JSONObject ...

  4. drf模块及源码

    drf中的APIView请求生命周期 APIView的as_view(局部禁用csrf) => 调用父类view中的as_view返回view()方法 => 自己的类调用自己的dispat ...

  5. 在Linux系统下进入MySql数据库进行操作

    例:   ---- 1.进入mysql数据库 root@test:/home# mysql -uroot -proot   <uroot是用户名,proot是密码> 2.查询所有的库 my ...

  6. 洛谷P1390 公约数的和 [2017年6月计划 数论12]

    P1390 公约数的和 题目描述 有一天,TIBBAR和LXL比赛谁先算出1~N这N个数中每任意两个不同的数的最大公约数的和.LXL还在敲一个复杂而冗长的程序,争取能在100s内出解.而TIBBAR则 ...

  7. 【Codeforces Round #430 (Div. 2) C】Ilya And The Tree

    [链接]点击打开链接 [题意] 给你一棵n个点的树,每个点的美丽值定义为根节点到这个点的路径上的所有权值的gcd. 现在,假设对于每一个点,在计算美丽值的时候,你可以将某一个点的权值置为0的话. 问你 ...

  8. 【python之路13】python的深浅拷贝

    深浅拷贝 一.数字和字符串 对于 数字 和 字符串 而言,赋值.浅拷贝和深拷贝无意义,因为其永远指向同一个内存地址. 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 impor ...

  9. Django独有报错的原因和解决

    RuntimeError at /login You called this URL via POST, but the URL doesn't end in a slash and you have ...

  10. 【BZOJ2809】【APIO2012】dispatching

    左偏树. 每个子节点维护大根堆,遍历一个儿子就往自己合并,合并发现钱不够了就删除队顶. //Achen #include<algorithm> #include<iostream&g ...