【原】javascript笔记之this用法
javascript中的this学习起来相对复杂,最近花了点时间研究,总结起来大概这只有5种情况,相信只要熟悉这5种用法,基本是可以解决所有的this问题,文本不介绍this设计原理,只介绍用法,阅读本文,你需要了解javascript执行上下文环境,博主写这种文章的目的,主要还是给自己做下笔记,后续也会输出javascript的学习笔记。
全局代码中的this & 调用普通函数中的this & this作为对象的属性值
全局代码中的this ,永远是window。
//全局环境下,this永远是window。
console.info(this === window);// true //定义全局对象的属性
this.cat = '猫'; // global.cat = '猫'
console.info(cat); // 猫 //给一个无标示符变量赋值
dog = '狗';
console.info(this.dog); // '狗' //通过变量声明
var bird = '鸟';
console.info(this.bird); // '鸟'
调用普通函数中的this,永远是window。
function fn1() {
this.cat = '包子'
console.info(this);//Window {postMessage: ƒ, blur: ƒ, focus: ƒ, close: ƒ, frames: Window, …}
console.info(this.cat);//包子
}
fn1()
this作为对象的属性值,永远是window。
let obj = {
cat : '猫宝宝',
cat_father : this,
cat_self : this.cat
}
console.info(obj.cat_father);//Window {postMessage: ƒ, blur: ƒ, focus: ƒ, close: ƒ, frames: Window, …}
console.info(obj.cat_self);//undefined
其它注意:任何时候不能改变this的值。
//报错,任何时候不能改变this的值
this = '鸟'
构造函数中this
函数作为构造函数用,那么其中的this就代表它即将new出来的对象。
function Fn2() {
this.cat = '包子',
this.dog = '饺子'
console.info(this);//Fn2 {cat: "包子", dog: "饺子"}
}
let fn2 = new Fn2();
console.info(fn2.cat);//包子
console.info(fn2.dog);//饺子
原型链中this
在构造函数的prototype中,this代表函数即将new出来的对象。
function Fn3() {
this.cat = '包子'
}
Fn3.prototype.getItem = function(){
return this.cat;
}
let fn3 = new Fn3();
console.info(fn3.getItem());//包子
其实,不仅仅是构造函数的prototype,即便是在整个原型链中,this代表的也都是当前对象的值。
函数作为对象的一个属性被调用
函数作为对象的一个属性被调用,函数中的this指向调用它的对象,加深红色的这句话非常关键。
let obj = {
cat : '猫宝宝',
fn : function(){
console.info(this === obj);//true
console.info(this.cat);//猫宝宝
}
}
obj.fn()
又如小程序中,使用Page(Object) 函数用来注册一个页面,接受一个 Object 类型参数,那么this指向该Object
Page({
data: {
version: '1.0.1',
cat:'张老板他妹'
},
onLoad: function () {
console.info(this.data);//{version: "1.0.1", cat: "张老板他妹"}
}
})
在函数中this到底取何值,是在函数真正被调用执行的时候确定的,函数定义的时候确定不了。因为this的取值是执行上下文环境的一部分,每次调用函数,都会产生一个新的执行上下文环境,但最终函数中的this指向调用它的对象
let obj = {
cat: '大猫',
fn: function () {
console.info(this.cat)
}
};
//fn函数作为obj对象的一个属性被调用,在obj环境中执行,函数中的this指向该对象
obj.fn() // 大猫
var fn_new = obj.fn;
var cat = '小猫';//全局环境的cat
//fn函数赋值给变量fn_new的时候并没有执行,此时this指向window,那么执行fn_new()时,this.cat对应取值为window.cat
fn_new() // 小猫
再来一个例子,对象的中嵌套子对象,子对象的属性值为函数,函数被子对象调用,那么函数中的this指向子对象,也就是函数中的this指向调用它的对象)
let obj = {
cat: '大猫',
obj_in: {
fn: function () {
console.info(this.cat) //undefined
}
}
}
//fn函数是被obj_in对象所调用,所以this指向的也obj_in对象
obj.obj_in.fn();
最后一个例子,对象的属性为函数,函数中嵌套函数,this放在嵌套函数中的情况
let obj = {
cat : '猫',
fn : function(){
function fn_in(){
console.info(this);//Window {postMessage: ƒ, blur: ƒ, focus: ƒ, close: ƒ, frames: Window, …}
console.info(this.cat);//undefined
}
//执行fn函数后,嵌套函数fn_in在fn环境中执行,回到文章中说的第一种情况,此时fn_in是普通函数,则它的this指向window
fn_in();
}
}
obj.fn()
函数用call或apply或bind调用
当一个函数被call和apply调用时,this的值就取传入的对象的值。
let obj1 = {
baobao : '猫'
}
let obj2 = {
baobao : '羊'
}
let obj3 = {
baobao : '鹅'
}
let fn3 = function(){
console.info(this.baobao);
}
fn3.apply(obj1);//猫
fn3.call(obj2);//羊
fn3.bind(obj3)();//鹅
同函数作为对象的一个属性被调用一样,函数fn4_in是在obj.fn4内部定义的,所以它仍然是一个普通的函数,this仍然指向window。
let obj3 = {
baobao : '猫'
}
let fn4 = function(){
function fn4_in(){
console.info(this.baobao);//undefined
}
fn4_in()
}
fn4.apply(obj3);
参考资料
http://www.ruanyifeng.com/blog/2018/06/javascript-this.html
http://www.cnblogs.com/TomXu/archive/2012/01/17/2310479.html
http://www.cnblogs.com/wangfupeng1988/p/3988422.html
【原】javascript笔记之this用法的更多相关文章
- 【原】javascript笔记之Array方法forEach&map&filter&some&every&reduce&reduceRight
做前端有多年了,看过不少技术文章,学了新的技术,但更新迭代快的大前端,庞大的知识库,很多学过就忘记了,特别在项目紧急的条件下,哪怕心中隐隐约约有学过一个方法,但会下意识的使用旧的方法去解决,多年前ES ...
- [Effective JavaScript 笔记]第3章:使用函数--个人总结
前言 这一章把平时会用到,但不会深究的知识点,分开细化地讲解了.里面很多内容在高3等基础内容里,也有很多讲到.但由于本身书籍的篇幅较大,很容易忽视对应的小知识点.这章里的许多小提示都很有帮助,特别是在 ...
- jQuery学习笔记之Ajax用法详解
这篇文章主要介绍了jQuery学习笔记之Ajax用法,结合实例形式较为详细的分析总结了jQuery中ajax的相关使用技巧,包括ajax请求.载入.处理.传递等,需要的朋友可以参考下 本文实例讲述了j ...
- javascript confirm()函数的用法
javascript confirm()函数的用法 confirm():确认消息对话框.用于允许用户做选择的动作.弹出的对话框中包含一确定按钮和一取消按钮. confirm(str) 参数说明: st ...
- [Effective JavaScript 笔记] 第4条:原始类型优于封闭对象
js有5种原始值类型:布尔值.数字.字符串.null和undefined. 用typeof检测一下: typeof true; //"boolean" typeof 2; //&q ...
- [Effective JavaScript 笔记] 第5条:避免对混合类型使用==运算符
“1.0e0”=={valueOf:function(){return true;}} 是值是多少? 这两个完全不同的值使用==运算符是相等的.为什么呢?请看<[Effective JavaSc ...
- [Effective JavaScript 笔记]第27条:使用闭包而不是字符串来封装代码
函数是一种将代码作为数据结构存储的便利方式,代码之后可以被执行.这使得富有表现力的高阶函数抽象如map和forEach成为可能.它也是js异步I/O方法的核心.与此同时,也可以将代码表示为字符串的形式 ...
- [Effective JavaScript 笔记]第28条:不要信赖函数对象的toString方法
js函数有一个非凡的特性,即将其源代码重现为字符串的能力. (function(x){ return x+1 }).toString();//"function (x){ return x+ ...
- javascript中的继承用法
本文实例汇总了javascript关于继承的用法,希望本文所述对大家的javascript程序设计有所帮助.分享给大家供大家参考.具体如下:代码如下: /** * 实现子类继承父类,但不会产生多余的属 ...
随机推荐
- [Java算法分析与设计]--顺序栈的实现
在程序的世界,栈的应用是相当广泛的.其后进先出的特性,我们可以应用到诸如计算.遍历.代码格式校对等各个方面.但是你知道栈的底层是怎么实现的吗?现在跟随本篇文章我们来一睹它的庐山真面目吧. 首先我们先定 ...
- 前端工程化(二)---webpack配置
导航 前端工程化(一)---工程基础目录搭建 前端工程化(二)---webpack配置 前端工程化(三)---Vue的开发模式 前端工程化(四)---helloWord 继续上一遍的配置,本节主要记录 ...
- 初探Margin负值(转)
相对而言,margin 负值的使用机率在布局中似乎很少,但是我相信一旦你开始掌握就会着迷,接下来我们看看关于margin负值的一些资料: 它是一个有效的属性,至少w3c中明确描述如下:”Negativ ...
- java定时任务调度-Timer(1)
一.定义 有且仅有一个后台线程对多个业务线程进行定时定频率的调度 二. Timer ----> Timer Task (中有run();方法) 通过 new Timer().schedul ...
- VMware Workstation 的安装和使用
https://blog.csdn.net/lamp_yang_3533/article/details/53136474 VMware Workstation 是一个虚拟PC的软件,利用VMwa ...
- mysql left join 左连接查询关联n多张表
left join 左连接即以左表为基准,显示坐标所有的行,右表与左表关联的数据会显示,不关联的则不显示.关键字为left join on. **基本用法如下: select table a left ...
- 从JVM内存管理的角度谈谈JAVA类的静态方法和静态属性
在JVM中,内存分为两个部分,Stack(栈)和Heap(堆),这里,我们从JVM的内存管理原理的角度来认识Stack和Heap,并通过这些原理认清Java中静态方法和静态属性的问题. 一般,JVM的 ...
- Creating your own auto-configuration
44. Creating your own auto-configuration If you work in a company that develops shared libraries, or ...
- sql server 高可用故障转移(完结)
安装完二个sql server 节点后,对外的虚拟ip是192.168.2.105 测试将sql server转到另一节点 转移后连接sql 虚拟ip 测试 通过windows日志查看远行状态 总结 ...
- VMWare Workstation虚拟机 安装Centos7 图文指南
本篇博文将讲述如何一步一步在VMWare Workstation 中安装Centos 7 1. 准备工作 VMWare Workstation Centos7 镜像 VMWare Workstatio ...