前言

  this是函数中的隐形参数,它绑定的值取决于函数的调用位置。

this的定义

  《你不知道的js》中是这样说的:是函数体内的隐式参数,this就是记录函数调用上下文的一个属性。可以在函数体中使用this引用函数的调用上下文。this的绑定关注的是函数的调用位置。

  调用位置:是函数在调用时的位置,区别于函数声明时的位置。

this的绑定规则

  this绑定方式取决于函数的调用位置,根据不同的场景,有四中绑定方式(默认、隐式、显示、构造调用中的绑定)。多种情况同时存在时,依靠的是这四中规则的优先级别采用相应的绑定。

  默认绑定

    当函数不使用任何修饰时直接调用。无论在哪里调用,它指向的都是全局对象(浏览器环境下指向的是window)。注:严格模式下this会绑定到undefined。

function run(){
console.log(this); //window
console.log("running");
function speak(){
console.log("speak");
console.log(this); // window
}
return speak;
}
run()();

  隐形绑定

    当函数拥有上下文对象的时候,隐式绑定规则会把函数调用中的this绑定到这个上下文对象。

// 对象的属性不管以任何形式拥有函数的引用时,通过对象属性调用这个函数时,这个函数当前调用上下文就是这个对象。
function foo(){
console.log(this.a);
}
var obj = {
a:1,
foo:foo
}
obj.foo(); // 1
// 直接上下文:只有最后一层调用影响this的绑定

function foo(){
  console.log(this.a);
}
var obj1 = {
  a:1,
  foo:foo
}
var obj2 = {
  a:2,
  obj1:obj1
}
obj2.obj1.foo(); // 1

  显示绑定

    直接指定this的绑定对象就叫做显示绑定。js中所有函数都从Function.prototype上继承了call和apply这两个方法。他们都是用来显示指定函数执行时其中的this的。

    call和apply的异同:

      目的相同:都用来指定函数执行时其中的this。

      参数个数和类型不同:apply接收两个参数,第一个是本次执行要绑定在函数this上的值,第二个是一个数组。call接收的是一个参数列表,参数需要一个一个的传,他会把第一个参数绑定到本次执行函数的this上去。

      参数的处理:apply会把第二个参数(Array类型)展开后依次传到本次执行的函数中去。call是取第一绑定到this,再把后面的参数依次传入本次执行函数。

注:argument是函数体中的另一个隐式参数。可以在函数体内拿到函数运行时,接收到的实参。它是一个伪数组,其中的元素和实参一 一对应
// 以下输出结果出自chrome
function test(){
console.log(test.arguments);
}
// apply 数组[1,2,3]被展开分成三个参数 依次传入 test。
test.apply({},[1,2,3]); // Arguments(3) [1, 2, 3, callee: ƒ, Symbol(Symbol.iterator): ƒ]
// call 数组[1,2,3] 作为一个参数传进了test中
test.call({},[1,2,3]); // Arguments [Array(3), callee: ƒ, Symbol(Symbol.iterator): ƒ]

  常用操作

// 使用call调用数组map方法。 
var arr = [1,2,3,4];
Array.prototype.map.call(arr,function(item){
console.log(item); // 分别输出 1 2 3 4
})

new绑定

  使用new 操作符调用函数叫做构造调用。函数做构造调用时,其中this的绑定。函数做构造调用时其中有四个默认操作。

  1,创建一个新的对象。

  2,这个对象会被执行原型链接(添加到构造函数的原型链的末尾)。

  3,把当前正在做构造调用的函数中的this绑定到新创建的对象。

  3,如果函数没有显式的返回其他对象,那么默认返回这个新建的对象。

// 对foo进行构造调用创建一个新的对象
function foo(a){
this.a = a;
}
var bar = new foo(2);
console.log(bar.a);

  js中的构造函数

    1,只是使用new操作符调用的普通函数。

    2,不属于某个类,也不会实例化某个类。

    3,所有函数使用new 进行调用时都被称作构造调用。

    4,js中不存在面向类语言意义上的构造函数,只有基于new操作符号的构造调用。

js函数—隐形参数this的更多相关文章

  1. js函数的参数

    js函数的参数: js是弱类型的编程语言,调用函数时既不在乎函数的参数,也不在意参数的类型 即便你定义的函数值接受两个参数,在调用这个函数时也未必一定要是两个参数.可以传递一个.三个甚至不传递参数,而 ...

  2. js函数定义 参数只要写名称就可以了

    js函数定义  参数只要写名称就可以了 以下为标准: function add(type)  { } 不要写成下面这个样子 function add(var type)  { } 哎 妹的  老何ja ...

  3. js函数中参数的传递

    数据类型 在 javascript 中数据类型可以分为两类: 基本类型值 primitive type,比如Undefined,Null,Boolean,Number,String. 引用类型值,也就 ...

  4. JS函数的参数对象arguments在严格模式下的限制

    在JS中,传入的函数的参数个数可以与定义函数的个数不一致,那么对于传入的实参的引用,则是arguments对象.然而改对象在严格模式和非严格模式下是由区分的: 1 在严格模式下arguments作为了 ...

  5. js 函数作为参数+接受任意数量参数

    javascript中的函数是“复合数据类型”,又成为“引用类型”.引用类型的变量指向存储单元中存放的是它们的实际存放地址.函数名是对函数的一种引用.var a=max_num ;a()就可以调用fu ...

  6. Node.js函数介绍(参数为一个函数)

    在JavaScript中,一个函数可以作为另一个函数的参数.我们可以先定义一个函数,然后传递,也可以在传递参数的地方直接定义函数. Node.js中函数的使用与Javascript类似,举例来说,你可 ...

  7. js函数定义参数个数和实际传入参数的对比

    因为js是一种弱类型的编程语言,对数据类型的要求没有其他编程语言的要求严格,所以在定义函数的时候不需要像java和C#一样对其传入参数的类型进行定义.那么传入参数的个数有没有影响呢?今天小猪就做了个实 ...

  8. JS函数 有参数的函数 参数可以多个,根据需要增减参数个数。参数之间用(逗号,)隔开

    有参数的函数 上节中add2()函数不能实现任意指定两数相加.其实,定义函数还可以如下格式: function 函数名(参数1,参数2) { 函数代码 } 注意:参数可以多个,根据需要增减参数个数.参 ...

  9. JS函数传递参数是是按值传递

    JavaScript在传参的时候只有一种传递方法那就是按值传递(来自红宝书第四版本) 函数在传递参数的时候会把实参的值拷贝过来一份,而基础类型数据值是存在内存中,在拷贝的时候会复制出来一份,而引用类型 ...

随机推荐

  1. Linux 查看并删除.svn目录

    1. find . -type d -name ".svn"|xargs rm -rf

  2. xgboost算法原理

    XGBoost是2014年3月陈天奇博士提出的,是基于CART树的一种boosting算法,XGBoost使用CART树有两点原因:对于分类问题,CART树的叶子结点对应的值是一个实际的分数,而非一个 ...

  3. <opengl>使用glu绘制二次曲面

    绘制二次曲面通常要以下四步:   1.首先我们创建一个二次方程状态对象 GLUquadricObj *m_pObj;    //保存绘图模式.法线模式.法线朝向.纹理等信息 //创建二次方程状态对象 ...

  4. LeetCode 426. Convert Binary Search Tree to Sorted Doubly Linked List

    原题链接在这里:https://leetcode.com/problems/convert-binary-search-tree-to-sorted-doubly-linked-list/ 题目: C ...

  5. 「51Nod1639」绑鞋带(概率

    1639 绑鞋带  基准时间限制:1 秒 空间限制:131072 KB 分值: 20 难度:3级算法题  收藏  关注 有n根鞋带混在一起,现在重复n次以下操作:随机抽出两个鞋带头,把它们绑在一起.可 ...

  6. codevs 1531山峰

    传送门 1531 山峰  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond   题目描述 Description Rocky山脉有n个山峰,一字排开,从西向东 ...

  7. bzoj 1004 Cards & poj 2409 Let it Bead —— 置换群

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1004 关于置换群:https://www.cnblogs.com/nietzsche-oie ...

  8. [原创]如何解决IE10下CkEditor报 --- SCRIPT5007: 无法获取未定义或 null 引用的属性“toLowerCase”

    如何解决IE10下CkEditor报 --- SCRIPT5007: 无法获取未定义或 null 引用的属性“toLowerCase” 错误 如果你的IE是IE10,且不是运行在IE的兼容模式你也许会 ...

  9. TFS 备注

    1,更改任何文件, 先checkout, 再继续更改. 2. 更新sln时, 一定要更新include文件 3. 每次提交代码放到shelf上, 自己本地建立2个workspace, 来进行coder ...

  10. js避免命名冲突

    [1]工程师甲编写功能A var a = 1; var b = 2; alert(a+b); [2]工程师乙添加新功能B var a = 2; var b = 1; alert(a-b); [3]上一 ...