call和apply用途与使用方法
当你看代码时,经常会看到以下情形:(在这个博客里面,参数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); 但是为什么不直接这样写呢?
先看一段代码的执行结果:
var str = "stringstr";
var num = 12;
var arr = [1, 2];
var obj = {name: "solid"}; console.log(str.toString()); // stringstr
console.log(num.toString()); //
console.log(arr.toString()); // 1,2
console.log(obj.toString()); // [object Object] console.log(Object.prototype.toString.call(str)); // [object String]
console.log(Object.prototype.toString.call(num)); // [object Number]
console.log(Object.prototype.toString.call(arr)); // [object Array]
console.log(Object.prototype.toString.call(obj)); // [object Object]
如果你调用Object.prototype.toString()函数是不是可以查看变量的类型啊,进而进行别的逻辑。
在js中,基本上所有的变量都是从Object继承来的,但是他们又都改写了toString方法,所以调用自身的toString方法会得到重写后的函数的结果。有的并没有什么实际的用处,如字符串的toString方法。
下面看点实用的引用吧:
// A程序员写了一个函数
function Person(name, age, sex){
this.name = name;
this.age = age;
this.sex = sex;
}
// 现在一个对象里,你的需求里也需要这三个参数
var zhangsan = {};
// 你就可以这样借用他的函数为你效力
Person.call(zhangsan, "张三", 18, "男");
console.log(zhangsan); // {name: "张三", age: 18, sex: "男"}
// 你可能还需要其他的信息,再写就可以了
function say(){
this.say = function (){
console.log(this.name + "具有说话的能力");
}
}
say.call(zhangsan);
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的参数是一个数组。例子具体见下面的代码
Person.call(zhangsan, "张三", 18, "男");
Person.apply(zhangsan, ["张三", 18, "男"]);
// 这两句是等价的
apply什么时候用呢?看下面的例子把
// 已知一个数组,求数组中的最大值
var numArr = [12, 21, 9, 18, 100, 0];
var max = Math.max.apply(null, numArr);
console.log(max); // 100 // 下面这个求最大值的函数你可以直接收藏使用了。
function max(){
return Math.max.apply(null, arguments);
}
console.log(max(12, 21, 9, 18, 100, 0)); // 100
call和apply用途与使用方法的更多相关文章
- apply()和call()的方法
apply()和call()的方法的区别 参考文档https://www.cnblogs.com/lengyuehuahun/p/5643625.html 一直都没太明白apply()与call()的 ...
- JavaScript中的apply()方法和call()方法使用介绍
1.每个函数都包含两个非继承而来的方法:apply()和call(). 2.他们的用途相同,都是在特定的作用域中调用函数. 3.接收参数方面不同,apply()接收两个参数,一个是函数运行的作用域(t ...
- JavaScript的apply()方法和call()方法
1 <script type="text/javascript"> 2 /*定义一个人类*/ 3 function Person(name,age) 4 { 5 thi ...
- 利用apply和arguments复用方法
首先,有个单例对象,它上面挂了很多静态工具方法.其中有一个是each,用来遍历数组或对象. var nativeForEach = [].forEach var nativeMap = [].map ...
- 彻底理解了call()方法,apply()方法和bind()方法
javascript中的每一个作用域中都有一个this对象,它代表的是调用函数的对象.在全局作用域中,this代表的是全局对象(在web浏览器中指的是window).如果包含this的函数是一个对象的 ...
- apply方法和call方法。函数属性与方法。
每个函数都有length属性哥prototype属性. length属性表示的是函数接入参数的个数 在es引用类型语言中,prototype是保存它们所有实例方法的真正所在.换句话来说,类似于toSt ...
- AngularJs $scope 里面的$apply 方法和$watch方法
Angular $scope 里面的$apply 方法 Scope提供$apply方法传播Model变化 <!DOCTYPE html> <html> <head> ...
- 【JavaScript】call和apply区别及使用方法
一.方法的定义call方法: 语法:fun.call(thisArg[, arg1[, arg2[, ...]]])定义:调用一个对象的一个方法,以另一个对象替换当前对象.说明:call 方法可以用来 ...
- apply方法和call方法的详解2
1.apply和call的区别在哪里 2.什么情况下用apply,什么情况下用call 3.apply的其他巧妙用法(一般在什么情况下可以使用apply) 我首先从网上查到关于apply和call的定 ...
随机推荐
- 访问树中的所有元素(DOM)
创建一个函数,给定页面上的DOM元素,将访问元素本身及其所有后代(而不仅仅是它的直接子代).对于访问的每个元素,函数应将该元素传递给提供的回调函数. 函数的参数应该是: 一个DOM元素 一个回调函数( ...
- Hackerrank--Emma and sum of products (FFT)
题目链接 Emma is really fond of integers and loves playing with them. Her friends were jealous, and to t ...
- springMVC原理解析
1:SpringMVC运行原理 2:工作流程 (1)客户端(浏览器)发送请求,直接请求到DispatcherServlet. (2)DispatcherServlet根据请求信息调用HandlerMa ...
- vue里图片压缩上传组件
//单图上传 <template> <div> <div class="uploader" v-if='!dwimg'> <van-upl ...
- redis范围查询应用 数据库 数据库学习 Redis redis范围查询的方法
redis范围查询应用. 需求 根据IP找到对应的城市 原来的解决方案 oracle表(ip_country): 查询IP对应的城市: 1.把a.b.c.d这样格式的IP转为一个数字,例如为把210. ...
- golang之Sprintf函数
- Spring表达式语言:SpEl
概念: 是一个支持运行时查询和操作的对象图的强大的表达式语言. 语法类似于EL:SpEl使用#{ ...}作为定界符,所有在大括号中的 字符都将被认为是SpEl SpEl为bean的属性进行动态赋值提 ...
- 给大家分享一张CSS选择器优选级图谱 !
- Spring → 03:核心机制
一.控制反转 1.1.控制反转的概念 (1).Inverse of Controller被称为控制反转或反向控制,其实真正体现的是“控制转移”.(2).所谓的控制指的是负责对象关系的指定.对象创建.初 ...
- Directx11教程(31) 纹理映射(1)
原文:Directx11教程(31) 纹理映射(1) 在前面的例子中,我们要么是直接给顶点赋颜色值,要么是在顶点属性中设置Diffuse和Specular系数,从而根据光照参数计算得到 ...