前言

在《javascript 之执行环境-08》文中说到,当JavaScript代码执行一段可执行代码时,会创建对应的执行上下文(execution context)。对于每个执行上下文,都有三个重要属性:

  • 变量对象(Variable object,VO)
  • 作用域链(Scope chain)
  • this

  JavaScript中的this跟其他语言有些不一样,比如Java .net语言中的this是在代码的执行阶段是不可变的,而JavaScript的this是在调用阶段进行绑定。也因为这一性质给了this很大的灵活性,即当函数在不同的调用方式下都可能会导致this的值不同;

定义

  this 对象是在运行时基于函数的执行环境绑定的,跟函数的调用位置有关而不是声明的位置;可以理解为this是在函数调用阶段绑定,也就是执行上下文创建的阶段进行赋值,保存在变量对象中;

四种绑定规则

new 构造函数绑定,this指向新创建的对象

 function createPerson(){
return new person();
}
function person() {
this.name = "Joel";
this.say=function(){
console.log(p)
console.log('hello' + this.name)
}
}
var p = new person();
p.say();
console.log(p.name);//Joel
console.log('开始')
var t=createPerson();
t.say();

不管是直接new 还是通过function createPerson 函数返回的对象,this 都是指向了新创建出来的对象;

显示绑定,this指向传进去的对象

     //call() apply() bind() 显示绑定
window.color = "red";
var Obj = { color: "blue" };
function sayColor() {
console.log(this.color);
}
sayColor();// red this window
sayColor.call(this);//red
sayColor.call(window);//red
sayColor.call(Obj);// blue 把this指针改为对象Obj
sayColor.apply(Obj);//blue 把this指针改为对象Obj
sayColor.bind(Obj)();//blue 把this指针改为对象Obj

如果你把null/undefined作为this的绑定对象传入call/apply/bind,这些值在调用时会被忽略,实际用的是默认的绑定规则;

   function foo() {
console.log(this.a)
}
var a=2;
foo.call(null);//
foo.call(undefined);//

隐士绑定

以对象的方法形式调用,this指向当前这个对象

    function foo(){
console.log(this)//{a: 2, name: "Joel", foo: ƒ}
console.log(this.a)//
}
var obj={
a:2,
name:'Joel',
foo:foo
};
obj.foo();

这时this指向当前obj对象,但是如果换种写法会造成this 丢失问题。

     function foo(){
console.log(this)//{a: 2, name: "Joel", foo: ƒ}
console.log(this.a)//
}
var obj={
a:2,
name:'Joel',
foo:foo
};
obj.foo();
//this 丢失的问题
var t= obj.foo;
t(); //window undefined

变量t此时保存的是函数的引用跟obj已经没有关系,所以此时this指向window。

默认绑定 严格模式下this 绑定到undefined,否则绑定到全局对象 window

   function foo(){
console.log(this)//window
console.log(this.a)//Joel
}
var a='Joel';
foo(); //严格模式
function fo(){
'use strict' //严格模式
console.log(this)//undefined
console.log(this.b)//报错 Cannot read property 'b' of undefined
}
var b='Joel';
fo();

以上是基本的this绑定规则,其中new、显示绑定很容易判断,其中比较容易错的是容易把默认绑定误认为是隐士绑定 如匿名函数、闭包、函数当做参数等;

独立调用:this 指向window

   var name='Joel',age=12;
function say(){
function say1(){
console.log(this.name);//window
function say2(){
name+='-l'
console.log(this.name);//window
}
say2()
}
say1();
}
say(); //匿名函数
(function(){
console.log(this.name)
})()

function 当做参数传递其实跟独立调用一样的原理

 function foo() {
console.log(this)//window
console.log(this.a)//oops global
}
function doFoo(fn) {
console.log(this);//window
fn();//类似与 foo()
}
var obj = {
a: 2,
foo: foo
}
var a = 'oops global';
doFoo(obj.foo);

同理setTimeout 也是一样

  var obj = {
a: 2,
foo: function() {
console.log(this);
},
foo2: function() {
console.log(this); //this 指向 obj
setTimeout(this.foo, 1000); // this 指向 window
}
}
var a = 'oops global';
obj.foo2();

闭包中的this

     var name = "Joel";
var obj = {
name: "My object",
getName: function() {
// var that = this; // 将getNameFunc()的this保存在that变量中
return function() {
return this.name;
};
}
}
console.log(obj.getName()()); // "Joel"

这里的虽然是对象的方法形式调用obj.getName(),在getName中的this是指向obj,但是返回的匿名函数中的this 为什么是window呢?

把最后的一句拆成两个步骤执行:

 var t=obj.getName();
t();

是不是有点像在独立调用呢?如果需要访问obj中的name,只需要把this对象缓存起来,在匿名函数中访问即可,把var that=this;去掉注释即可;

总结

  • this 是变量对象的一个属性,是在调用时被绑定的,跟函数的调用位置有关而不是声明的位置;
  • 找到调用位置,根据绑定规则来分析this 绑定;
  • 默认绑定严格模式下this 绑定到undefined,否则绑定到全局对象 window;

思考题

     var length = 5;
var obj = {
foo: function (fn) {
console.log(this.length); // this => obj
fn(); // this => window
arguments[0](); // this => arguments
var bar = arguments[0];
bar(); // this => window
},
length: 10
}
var fn = function () {
console.log(this.length);
}
obj.foo(fn);
//10, 5, 1, 5

javascript 之this指针-11的更多相关文章

  1. 图说js中的this——深入理解javascript中this指针

    没搞错吧!js写了那么多年,this还是会搞错!没搞错,javascript就是回搞错! ………… 文章来源自——周陆军的个人网站:http://zhoulujun.cn/zhoulujun/html ...

  2. javascript中this指针

    看完此片文章豁然开朗,非常感谢.javascript技术难点(三)之this.new.apply和call详解 下面说一说自己的理解: this指针总是指向调用他的对象,其实我更愿意理解为:this指 ...

  3. javascript中this指针探讨

    javascript是一门类java语言有很多跟java相类似的特点,但也仅是类似而已,真正使用中还是有很大的差别.this指针常常让很多初学者抓狂,本人也曾为此困惑不解,查找过很多资料,今天在这里总 ...

  4. 原生JavaScript技巧大收集(11~20)-(终于又被我找到这篇文章了)

    11.原生JavaScript加入收藏夹 function AddFavorite(sURL, sTitle) { try { window.external.addFavorite(sURL, sT ...

  5. JavaScript——对this指针的新理解

    一直以来对this的理解只在可以用,会用,却没有去深究其本质.这次,借着<JavaScript The Good Parts>,作了一次深刻的理解.(所有调试都可以在控制台中看到,浏览器F ...

  6. javascript 误用this指针 的情况

    理解了this指针后,我们再来看看一些很容易误用this指针的情况. 示例1——内联式绑定Dom元素的事件处理函数 <script type="text/javascript" ...

  7. javascript中this指针的认识

    javascript中上下文环境就是this指针,即被调用函数所处的环境.这个上下文环境在大多数情况下指的是函数运行时封装这个函数的那个对象:当不通过任何对象单独调用一个函数时,上下文环境指的就是全局 ...

  8. Javascript图片预加载详解 分类: JavaScript HTML+CSS 2015-05-29 11:01 768人阅读 评论(0) 收藏

    预加载图片是提高用户体验的一个很好方法.图片预先加载到浏览器中,访问者便可顺利地在你的网站上冲浪,并享受到极快的加载速度.这对图片画廊及图片占据很大比例的网站来说十分有利,它保证了图片快速.无缝地发布 ...

  9. JavaScript的this指针到底指向哪?

    编程过程中,着实十分困扰this的指向性,经过查阅一番资料,终于搞清楚了,在这里总结一下,全文分为以下三个部分: 什么是this指针? this指针指向哪里? 何时使用this? 一 什么是this指 ...

随机推荐

  1. hibernate学习笔记之中的一个(JDBC回想-ORM规范)

    JDBC回想-ORM规范 JDBC操作步骤 注冊数据库驱动 Class.forName("JDBCDriverClass") 数据库 驱动程序类 来源 Access sun.jdb ...

  2. Material使用05 自定义主题、黑夜模式\白天模式切换

    需求: 1 不使用materil依赖内建的主题,使用自己创建的主题 2 利用自己创建的主题实现白天模式和黑夜模式 1 自定义主题 1.1 创建自定义主题文件 them.scss // 引入materi ...

  3. 让PIP源使用国内镜像,提升下载速度和安装成功率。

      对于Python开发用户来讲,PIP安装软件包是家常便饭.但国外的源下载速度实在太慢,浪费时间.而且经常出现下载后安装出错问题.所以把PIP安装源替换成国内镜像,可以大幅提升下载速度,还可以提高安 ...

  4. python filter map reduce

    filter(function, iterable): Construct a list from those elements of iterable for which function retu ...

  5. JavaScript:inherits

    网上一查,肯定搜索到继承的文章真心不少.我这里就只说一下自己常用的方式: 通常 在编写一个类的做法是,在构造函数里声明字段,在prototype里指定方法. //step1: 在子类的构造器里法里实例 ...

  6. iOS Swift 模块练习/swift基础学习

    SWIFT项目练习     SWIFT项目练习2 iOS Swift基础知识代码 推荐:Swift学习使用知识代码软件 0.swift中的宏定义(使用方法代替宏) 一.视图  +控件 1.UIImag ...

  7. iOS 设置视图背景的透明度

    p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 11.0px Menlo; color: #00afca } span.s1 { color: #fffff ...

  8. iOS: FFmpeg编译和使用 学习

    ffmpeg是一个多平台多媒体处理工具,处理视频和音频的功能非常强大.目前在网上搜到的iOS上使用FFMPEG的资料都比较陈旧,而FFMPEG更新迭代比较快: 且网上的讲解不够详细,对于初次接触FFM ...

  9. 每周.NET前沿技术文章摘要(2017-06-07)

    汇总国外.NET社区相关文章,覆盖.NET ,ASP.NET等内容: .NET .NET Core and .NET Framework Working Together, Or: The Magic ...

  10. ArcGIS API for JavaScript 4.2学习笔记[25] 官方第八章Analysis(空间查询)概览与解释

    开森,最关注的空间分析章节终于到了,在空间查询那节逻辑性的代码简直要命(呵呵,空间分析的代码也要命...). 上目录截图: [Geodesic buffers(GeometryEngine)] 使用G ...