javascript 之this指针-11
前言
在《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的更多相关文章
- 图说js中的this——深入理解javascript中this指针
没搞错吧!js写了那么多年,this还是会搞错!没搞错,javascript就是回搞错! ………… 文章来源自——周陆军的个人网站:http://zhoulujun.cn/zhoulujun/html ...
- javascript中this指针
看完此片文章豁然开朗,非常感谢.javascript技术难点(三)之this.new.apply和call详解 下面说一说自己的理解: this指针总是指向调用他的对象,其实我更愿意理解为:this指 ...
- javascript中this指针探讨
javascript是一门类java语言有很多跟java相类似的特点,但也仅是类似而已,真正使用中还是有很大的差别.this指针常常让很多初学者抓狂,本人也曾为此困惑不解,查找过很多资料,今天在这里总 ...
- 原生JavaScript技巧大收集(11~20)-(终于又被我找到这篇文章了)
11.原生JavaScript加入收藏夹 function AddFavorite(sURL, sTitle) { try { window.external.addFavorite(sURL, sT ...
- JavaScript——对this指针的新理解
一直以来对this的理解只在可以用,会用,却没有去深究其本质.这次,借着<JavaScript The Good Parts>,作了一次深刻的理解.(所有调试都可以在控制台中看到,浏览器F ...
- javascript 误用this指针 的情况
理解了this指针后,我们再来看看一些很容易误用this指针的情况. 示例1——内联式绑定Dom元素的事件处理函数 <script type="text/javascript" ...
- javascript中this指针的认识
javascript中上下文环境就是this指针,即被调用函数所处的环境.这个上下文环境在大多数情况下指的是函数运行时封装这个函数的那个对象:当不通过任何对象单独调用一个函数时,上下文环境指的就是全局 ...
- Javascript图片预加载详解 分类: JavaScript HTML+CSS 2015-05-29 11:01 768人阅读 评论(0) 收藏
预加载图片是提高用户体验的一个很好方法.图片预先加载到浏览器中,访问者便可顺利地在你的网站上冲浪,并享受到极快的加载速度.这对图片画廊及图片占据很大比例的网站来说十分有利,它保证了图片快速.无缝地发布 ...
- JavaScript的this指针到底指向哪?
编程过程中,着实十分困扰this的指向性,经过查阅一番资料,终于搞清楚了,在这里总结一下,全文分为以下三个部分: 什么是this指针? this指针指向哪里? 何时使用this? 一 什么是this指 ...
随机推荐
- linux进程标识符具体解释1
每一个进程都有一个实际用户标识符和一个实际组标识符,它们永远是启动该进程之用户的用户标识符和组标识符. 进程的有效用户标识符和有效组标识符或许更重要些,它们被用来确定一个用户是否能訪问某个确定的文件. ...
- PKI(公钥基础设施)基础知识笔记
数字签名 数字签名(又称公钥数字签名.电子签章)是一种类似写在纸上的普通的物理签名,可是使用了公钥加密领域的技术实现.用于鉴别数字信息的方法. 一套数字签名通常定义两种互补的运算.一个用于签名,还有一 ...
- Android多线程研究(2)——定时器
先来看一段代码: public static void main(String[] args) { new Timer().schedule(new TimerTask() { @Override p ...
- 深入理解计算机系统_3e 第四章家庭作业(部分) CS:APP3e chapter 4 homework
4.52以后的题目中的代码大多是书上的,如需使用请联系 randy.bryant@cs.cmu.edu 更新:关于编译Y86-64中遇到的问题,可以参考一下CS:APP3e 深入理解计算机系统_3e ...
- 体验mssql-cli
1. 背景 安装SQL Server on Linux之后,在命令行下使用sqlcmd,你会发现代码提示,语法高亮,甚至连多行复制都不支持,相比之下,MySQL的命令行客户端还好用多了.只做简单的命令 ...
- [转载]ArchLinux 添加 archlinuxcn 源 密钥错误
http://www.jianshu.com/p/8ed688b0a096 杜龙少 正在检查密钥环里的密钥 [######################] 100% 正在下载所需的密钥...... ...
- Android查缺补漏--ContentProvider的使用
ContentProvider (内容提供者)是一种共享型组件,可以为系统内应用于与应用之间提供访问接口. ContentProvide要想正常工作需要三个关键点: ContentProvider:对 ...
- 小白的Python之路 day1 表达式if ... else ,while循环,for循环
表达式if ... else 一.用户登陆验证 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 # 提示输入用户名和密码 # 验 ...
- ArcGIS API for JavaScript 4.2学习笔记[0] AJS4.2概述、新特性、未来产品线计划与AJS笔记目录
放着好好的成熟的AJS 3.19不学,为什么要去碰乳臭未干的AJS 4.2? 4.2全线基础学习请点击[直达] 4.3及更高版本的补充学习请关注我的博客. ArcGIS API for JavaScr ...
- 视觉SLAM
SLAM:Simultaneous Localization And Mapping.中文:同时定位与地图重建. 它是指搭载特定传感器的主体,在没有实验先验信息的情况下,于运动过程中建立环境的模型,同 ...