人们对于this的绑定常常有两个误解,一:指向函数本身,二:指向函数作用域。这两种想法都是错的,this并不指向函数本身,也不指向函数作用域。

function  foo(){
this.count++;
}
foo.count = 0;
for(var i = 0 ; i<5 ; i++){
foo();
}
alert( foo.count ); //

因为this不指向函数本身,所以foo.count的值依然是0。

function foo(){
var a = 1 ;
alert(this.a);
}
var a = 2;
foo(); //

因为this不指向函数作用域,所以输出的是2。

this实际上是在函数被调用时发生的绑定,它指向什么完全取决于函数在哪里被调用。

this的绑定一共有四种绑定:1:默认绑定(即没有明确的调用对象)

2:隐性绑定(即作为对象方法调用,this会被绑定到该对象)

3:显性绑定(使用apply()和call()调用,两个方法的第一个参数为一个对象,this被绑定到该对象)

4:new绑定(使用new来调用函数,会构造一个新对象,并且把this绑定到该对象)

一:默认绑定   (即没有明确的调用对象)   

function foo(){
var a = 1 ;
alert(this.a);
}
var a = 2;
foo(); // 2 (非严格模式下)

没有明确调用对象,this会被绑定到window对象,所以this.a就是window.a,即为2。不过这得在非严格模式下,只有在非严格模式下this才会被绑定到window对象,而在严格模式下,this被绑定到undefined。

 二:隐性绑定   (即作为对象方法调用,this会被绑定到该对象)

function  foo(){
var a = 1;
alert(this.a);
}
var obj = {
a:2,
foo:foo
};
obj.foo(); //

obj对象调用foo()函数,this被绑定到obj对象,所以输出了obj对象的a的值2。

对象属性引用链中只有最后一层会影响调用位置

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

obj1和obj2两个对象连续调用,this会被绑定到最后一个对象,即obj2,所以输出2

隐式丢失:被隐式绑定的函数会丢失绑定对象,有两种情况会丢失,一种为引用,另一种为回调函数。

         引用:

function  foo(){
alert(this.a);
}
var obj = {
a:1,
foo:foo
};
var bar = obj.foo;
var a = "global 1";
bar(); //global 1

bar是obj.foo的一个引用,实际上它引用的是foo函数本身,所以this被绑定到window对象,输出的是"global 1"

回调函数:

function    foo(){
alert(this.a);
}
function doFoo(fn){
fn();
}
var obj = {
a = 1,
foo:foo
}
var a = "global 1";
doFoo(obj.foo); //"global 1"

    调用回调函数的函数可能会修改this

 三:显性绑定(使用apply()和call()调用,两个方法的第一个参数为一个对象,this被绑定到该对象) 

function  foo(){
alert(this.a);
}
var obj = {
a:1
};
foo.call(obj); //

call()的参数若为空,默认调用window对象,若为一个原始值(字符串类型,布尔类型或者数字类型),则这个原始值会被转换成它的对象形式(new  String(),new Boolean() 或者 new Number()),这被称为“装箱”

硬绑定———显示绑定的一种变形

            优点:可以解决丢失绑定问题

缺点:硬绑定后不可能再修改它的this   

function  foo(){
alert(this.a);
}
var obj = {
a:2
};
var bar = function(){
foo.call(obj);
};
bar(); //
setTimeout(bar,100); //
bar.call(window); //

 ES5中提供了内置方法Function.prototype.bind

function foo(something){
alert(this.a, something);
return this.a + something;
}
var obj = {
a:2
};
var bar = foo.bind(obj);
var b = bar(3); // 2 3
alert(b); //

bind()会返回一个硬编码的新函数,它会把参数设置为this的上下文并调用原始函数

四:new绑定(使用new来调用函数,会构造一个新对象,并且把this绑定到该对象)

function   foo(a){
this.a = a;
}
var bar = new foo(2);
alert(bar.a); //

this绑定的四条规则的优先级: new绑定  >  显示绑定  >  隐式绑定   >  默认绑定

js中this的绑定的更多相关文章

  1. js中的前绑定和后绑定详解

    这篇文章详细介绍了js中的前绑定和后绑定,有需要的朋友可以参考一下 其主要意思就是看我有没有用过前绑定,即Dom树中的某些元素在还没有创建出来时,就指定该类型的元素一出生就应该拥有的某些事件.在实际开 ...

  2. js中this的绑定规则及优先级

    一.   this绑定规则 函数调用位置决定了this的绑定对象,必须找到正确的调用位置判断需要应用下面四条规则中的哪一条. 1.1 默认绑定 看下面代码: function foo() { cons ...

  3. js中同一个onclick绑定了两个js方法出现的问题

    问题: js中如果同一个onclick绑定了两个js方法问题,即 <li onclick="f1(),f2()"></li> 两个方法f1,f2中都分别有一 ...

  4. js中的事件绑定的三种方式

    1 直接在html标签中绑定 <button onclick = "show()"></button> 注意当你引用的js代码是包裹在window.onlo ...

  5. 关于js中对事件绑定与普通事件的理解

    普通事件指的是可以用来注册的事件: 事件绑定是指把事件注册到具体的元素之上. 通俗点说: 普通事件:给html元素添加一个特定的属性(如:onclick): 事件绑定:js代码中通过标记(id  ta ...

  6. JS 中的事件绑定、事件监听、事件委托

    事件绑定 要想让 JavaScript 对用户的操作作出响应,首先要对 DOM 元素绑定事件处理函数.所谓事件处理函数,就是处理用户操作的函数,不同的操作对应不同的名称. 在JavaScript中,有 ...

  7. js中传统事件绑定模拟现代事件处理

    大家都知道,IE中的现代事件绑定(attachEvent)与W3C标准的(addEventListener)相比存在很多问题, 例如:内存泄漏,重复添加事件并触发的时候是倒叙执行等. 下面是用传统事件 ...

  8. JS 中的事件绑定、事件监听、事件委托是什么?

    在JavaScript的学习中,我们经常会遇到JavaScript的事件机制,例如,事件绑定.事件监听.事件委托(事件代理)等.这些名词是什么意思呢,有什么作用呢? 事件绑定 要想让 JavaScri ...

  9. JS中的事件绑定,事件捕获,事件冒泡以及事件委托,兼容IE

    转载请注明出处:http://www.cnblogs.com/zhangmingze/p/4864367.html   ● 事件分为三个阶段:   事件捕获 -->  事件目标 -->   ...

随机推荐

  1. java反射 之 反射基础

    一.反射 反射:JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为 ...

  2. 【深入浅出Linux网络编程】 "开篇 -- 知其然,知其所以然"

    [深入浅出Linux网络编程]是一个连载博客,内容源于本人的工作经验,旨在给读者提供靠谱高效的学习途径,不必在零散的互联网资源中浪费精力,快速的掌握Linux网络编程. 连载包含4篇,会陆续编写发出, ...

  3. Android 对程序异常崩溃的捕捉

    转载博客:http://blog.csdn.net/i_lovefish/article/details/17719081 以下为异常捕捉处理代码: import java.io.BufferedRe ...

  4. java中Set,Map,Stack一些简单用法

    import java.util.Iterator; import java.util.Stack; import java.io.*; import java.util.Set; import ja ...

  5. 使用ikmv将jar转换为dll,在.NET中直接调用

    ikmv百科介绍 ikvm官网 使用ikvm前需要先安装java. ikvm的使用比较简单,到官网下载ikvm需要的组件,解压后在path中配置 %IKVM_HOME%\bin 在cmd中使用命令 i ...

  6. angular开发者吐槽react+redux的复杂:“一个demo证明你的开发效率低下”

    曾经看到一篇文章,写的是jquery开发者吐槽angular的复杂.作为一个angular开发者,我来吐槽一下react+redux的复杂. 例子 为了让大家看得舒服,我用最简单的一个demo来展示r ...

  7. 纯CSS打造忙碌光标

    我们经常会用到忙碌等待的情况,比如ajax等待回调,或者加载页面的时候,我们通常的做法就是放一张loading.gif图片,在那里转呀转. 以前是在pc上用,也没有觉得不妥,如今要在移动端用,问题就来 ...

  8. DDD 领域驱动设计-Value Object(值对象)如何使用 EF 进行正确映射

    写在前面 首先,这篇博文是用博客园新发布的 MarkDown编辑器 编写的,这也是我第一次使用,语法也不是很熟悉,但我觉得应该会很爽,博文后面再记录下用过的感受,这边就不多说. 阅读目录: 上一篇回顾 ...

  9. (十六)WebGIS中偏移补偿量引发的问题之探讨

    文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/. 1.背景 在上一章里讲解地图平移功能的实现时,我在最后提出了两个问题: ...

  10. 2.Java基础之Runtime对象

    毕向东老师Java基础学习笔记——Runtime对象 今天学习Java中的Runtime对象后,感觉这个对象对我们主要有以下几点用处. 1.使用java代码打开本地可执行文件,比如打开一个计算器. 2 ...