玩转Javascript this用法
在web项目中Javascript是一门必须要掌握的动态语言,基于Javascript的框架大多离不开不了最基础的Javascript的用法和原理。本文主要是总结一下Javascript中那万恶的this关键字。
开门见山,抛出一个观点:“Javascript中this永远是指向调用它的对象”。下面我会举3个最有代表性的例子来验证我的这个观点。
例一 对象方法调用
var x = 1;
function testThis(){
console.log(this.x);
} testThis(); // 这里声明了一个全局变量x,一个全局方法,这两个对象都绑定在Window上,所以当运行testThis()的结果就是取Window对象上的x成员,结果是1 var o={};
o.x = 5;
o.method = testThis; o.method(); // 此时,我们讲o对象的method指向了testThis, 当调用它的时候,this指向调用他的对象,这是x就是去o对象的x,结果是5
这个例子十分基础,也就是常见的对象调用方法的时候,方法里面的this就是指向调用他或者是拥有他的对象
例二 构造函数创建
var x = 2;
function test(){
this.x = 1;
} var o = new test();
console.log(o.x); // //这就是javascript中的构造函数,通过new来创建一个实例,这里取的值是绑定在对象o里面的x,所以是1 //下面这个例子是Angular中的Service,你可以直接理解成他会通过第二个参数new一个单例对象
app.service("MyService", function ($http, $modal) {
this.test = function() {
console.log(this);
test2();
} function test2() {
console.log(this)
}
} MyService.test() // 打印 MyService {test: function} 和 Window {} //这里log出来虽然test2()可以被test1()调用,但它其实并不属于Service, 所以如果test2里面调用this,就会出现常见的错误,哎呀妈呀,咋调不了自己的方法?!
这个例子我们项目中经常出现,而且很难解释清楚,就像还有人问我为什么controller中不直接使用this,而是要用$scope来绑定方法和变量,当然用this能够取代部分$scope,但是难免遇到this的上下文不同引起的一系列问题。这个问题的关键就是test2方法并不属于对象Service,但是由于在service闭包(closure)里面,他可以被Service调用,所以test2里面的this就不是指向Service,从而调用Service里面的其他方法就会报错。
例三 改变this指向
function test(){
console.log(this.x);
}
var o={};
o.x = 1;
o.m = test;
o.m.apply({x:5}); //
o.m.call() // undefined
//通过apply/call来指定调用函数的this作用上下文,都是指用参数对象来调用o对象的函数,默认参数是Global
其实通过这个例子,大家就已经可以看到this的指向是不确定的,this值在进入上下文时确定,并且在上下文运行期间永久不变。上面的例子改变this的上下文,导致两次结果不一致也是最好的证据。
最后
本文最大的作用就是如果看完本文你能够理解this为什么有时候会跟自己期望不一样,而且能得到一个避免这样问题发生的解决方案,那我的目的就达到了。很幸运你能看到这里,解决方案呈现给大家:就像Angular要将属性和方法给你提供一个$scope来绑定属性,也像coffeeScript会在对象一开头有这么个赋值“ _this = this”,我们在自己的js实践中不妨定义"val self = this",在之后的作用域里面用self来操作对象的属性,这是解决之道。
玩转Javascript this用法的更多相关文章
- 玩下Javascript
玩下Javascript 前言 好久没有更新博客了,也蛮少捣弄javascript,今儿看到一个题目,关于给你一个面板,你可以随意的在上面画矩形,可以移动和删除任意一个你创建的矩形,心血来潮搞着玩哈, ...
- javascript webstorm用法
javascript webstorm用法 一.什么是webstorm? WebStorm 是jetbrains公司旗下一款JavaScript 开发工具.被广大中国JS开发者誉为“We ...
- JavaScript this用法总结(**************************************)
JavaScript this用法总结 在JavaScript中,this关键字可以说是最复杂的机制之一.对this的作用机制缺乏比较深入的理解很容易在实际开发中出现问题. 1.this的作用 为什么 ...
- <a href="javascript:;">的用法说明
<a href="javascript:;">的用法说明 1.标签的 href 属性用于指定超链接目标的 URL,href 属性的值可以是任何有效文档的相对或绝对 UR ...
- 玩转JavaScript module pattern精髓
JavaScript module pattern是一种常见的javascript编码模式.这种模式本身很好理解,但是有很多高级用法还没有得到大家的注意.本文,我们将回顾这种设计模式,并且介绍一些高级 ...
- javascript:void(0)和javascript:;的用法
一.JavaScript:void(0) 我们经常会使用到 javascript:void(0) 这样的代码,那么在 JavaScript 中 javascript:void(0) 代表的是什么意思呢 ...
- 弱弱的玩下Javascript
前言 好久没有更新博客了,也蛮少捣弄javascript,今儿看到一个题目,关于给你一个面板,你可以随意的在上面画矩形,可以移动和删除任意一个你创建的矩形,心血来潮搞着玩哈,实现起来挺简单的,但这代码 ...
- JavaScript高级用法二之内置对象
综述 本篇的主要内容来自慕课网,内置对象,主要内容如下 1 什么是对象 2 Date 日期对象 3 返回/设置年份方法 4 返回星期方法 5 返回/设置时间方法 6 String 字符串对象 7 返回 ...
- Javascript typeof 用法
在js里用到数组,比如 多个名字相同的input, 若是动态生成的, 提交时就需要判断其是否是数组. if(document.mylist.length != "undefined" ...
随机推荐
- hibernate、struts、spring mvc的作用
Hibernate工作原理及为什么要用?原理:1.读取并解析配置文件2.读取并解析映射信息,创建SessionFactory3.打开Sesssion4.创建事务Transation5.持久化操作6.提 ...
- ERROR 1045 (28000): Access denied for user 'root'@'127.0.0.1' (using password: YES)
我的原因是在配置文件my.ini [mysqld]项,在其后加入了一句:skip-name-resolve 导致授权出现这个错误,把skip-name-resolve这项屏蔽了就好了. 场景2:对所有 ...
- 【c语言】推断一个数是奇偶数
// 推断一个数是奇偶数 #include <stdio.h> void judge_sd(int a) { if ((a & 1) == 0) { printf("是偶 ...
- Gray Code - 格雷码
基本概念 格雷码是一种准权码,具有一种反射特性和循环特性的单步自补码,它的循环.单步特性消除了随机取数时出现重大误差的可能,它的反射.自补特性使得求反非常方便.格雷码属于可靠性编码,是一种错误最小化的 ...
- 监视在input框中按下回车(enter) js实现
function getKey(event) { if (event.keyCode == 13) { alert("我是回车键"); } } <input type=&qu ...
- Tomcat 配置 项目 到tomcat目录外面 和 域名绑定访问(api接口、前端网站、后台管理网站)
先停止tomcat服务 1.进入apache-tomcat-7.0.68/conf/Catalina/localhost(如果之前还都没有启动过tomcat,是不会有此目录的,先启动一次再关闭,会自动 ...
- 使用存储过程将Oracle数据批量导出为多个csv文件
数据库有如下表结构: user_info ( user_id NUMBER primary key, user_name VARCHAR2(200) NOT NUL ...
- unity如何停止不用字符串方式开启协程的方法
通常我们知道开启协程用StartCoroutine("Method"); 停止协程用StopCoroutine("Method"); 如果我们想要终止所有的协程 ...
- 告诉你html5比普通html多了哪些东西?
- Oracle中的三种循环(For、While、Loop)
from:http://jingyan.baidu.com/article/c275f6ba38036ae33c756773.html GOTO用法,以下是SQL源码: DECLARE x numb ...