当前位置: 首页 > news >正文

【JavaScript】this 指向

1、this 指向谁

多数情况下,this 指向调用它所在方法的那个对象。即谁调的函数,this 就归谁。

当调用方法没有明确对象时,this 就指向全局对象。在浏览器中,指向 window;在 Node 中,指向 Global。(严格模式下,指向 undefined)

this 的指向是在调用时决定的,而不是在书写时决定的。这点和闭包恰恰相反。

2、区分 “声明位置” 与 “调用位置”

js 是词法作用域模型,对象或方法,其生命周期只和声明位置有关。

示例:

// 声明位置
var me = {name: 'xiaohong',hello: function() {console.log(`${this.name}`)}
}
var you = {name: 'xiaoming',hello: me.hello
}// 调用位置
me.hello() 	// xiaohong
you.hello() // xiaoming

两次调用的 this 也就分别指向了 me 和 you。

// 声明位置
var me = {name: 'xiaohong',hello: function() {console.log(`${this.name}`)}
}
var name = 'xiaoming'
var hello = me.hello 
// 调用位置
me.hello() 	// xiaohong
hello() 		// xiaoming

me.hello 赋值给变量 hello 时,实际上是在创建一个新的函数引用,这个引用指向 me 对象中的 hello 方法。但是,这个新的引用(即变量 hello)并不保留与 me 对象的任何关联或绑定。

调用 hello() 时,是在全局作用域中调用这个函数,而不是作为 me 对象的方法调用。因此,在这个调用中,this 的值不会指向 me 对象。

// 声明位置
var me = {name: 'xiaohong',hello: function() {console.log(`${this.name}`)}
}
var you = {name: 'xiaoming',hello: function() {var targetFunc = me.hellotargetFunc()}
}
var name = 'xiaosan'
// 调用位置
you.hello() // xiaosan

调用一个对象的方法时(例如 me.hello()),this 在该方法内部指向调用该方法的对象(在这个例子中是 me)。但是,将一个方法赋值给一个变量(例如 var targetFunc = me.hello),然后像普通函数那样调用这个变量(例如 targetFunc()),this 就不会再指向原来的对象了。相反,在非严格模式下,this 会默认指向全局对象(在浏览器中是 window),而在严格模式下,this 会是 undefined

3、普通函数的this指向总结

① 默认绑定:

  • 在全局中声明的变量和函数(默认指向Window);

  • 函数独立调用时(声明式函数、匿名函数 / 赋值式的方式、闭包)(都是指向Window);

  • 立即执行函数、 setTimeout、setInterval 指向 window

② 隐式绑定:

  • 对象调用(也就是谁调用就是指向谁,所以就是指向调用这个函数的对象)

(存在隐式丢失的问题:函数赋值和参数赋值的情况);

  • 绑定的事件处理函数(指向的是绑定事件处理函数的标签);

③ 显式绑定:

  • call / apply / bind (指向第一个参数)

④ new绑定:

  • 构造函数中的this指向实例化出来的对象;

4、箭头函数

箭头函数内部是没有this指向的,箭头函数的this指向父级作用域的this;如果没有,则this指向的就是Window。

箭头函数注意事项:

  • 使用了箭头函数,this就不是指向window,而是父级(指向是可变的)

  • 不能够使用arguments对象

  • 不能用作构造函数,这就是说不能够使用new命令,否则会抛出一个错误

      var Foo = () => {console.log(this);};var a = new Foo(); // 报错Foo is not a constructor
    
  • 不可以使用 yield 命令,因此箭头函数不能用作 Generator 函数

示例:

function foo() {console.log(this) // obj对象// 情况一// function test() {//     console.log(this) // Window// }// test()// 情况二// function test() {//     console.log(this) // obj对象// }// test.call(this)// 情况三var test = () => {console.log(this) // obj对象}test()
}
var obj = {a: 1,foo: foo
}
obj.foo()
const obj = {sayThis: () => {console.log(this);}
};
// 因为JavaScript没有块作用域,所以在定义sayThis的时候,里面的this就绑到window上去了
obj.sayThis(); 	// window 
const globalSay = obj.sayThis;
globalSay(); 		// window 浏览器中的global对象
const liEle = document.querySelectorAll('li');
liEle.forEach((item, key) => {// 箭头函数的this指向的是父级程序// forEach()的this指向windowconsole.log('打印1', this) // Windowitem.addEventListener('click', () => {console.log('打印2', this) // Window})
})
const button = document.getElementById('btn');
button.addEventListener('click', () => {console.log(this) // Window
})
function Cat(title) {this.title = title
}
Cat.prototype.sayName = () => {console.log(this) // Windowreturn this.title
}
const cat = new Cat('我是标题啊');
console.log(cat.sayName()) // undefined
function foo() {console.log(this) // obj对象var test = () => {console.log(this) // obj对象}return test
}
var obj = {a: 1,foo: foo
}
obj.foo()()
function foo1() {console.log(this) // obj对象
}
var foo2 = () => {console.log(this) // Window
}
var obj = {a: 1,foo1: foo1,foo2: foo2
}
obj.foo1()
obj.foo2()
function foo1() {console.log(this) // obj2对象
}
var foo2 = () => {console.log(this) // Window
}
var obj = {a: 1,foo1: foo1,foo2: foo2
}
var obj2 = {a: 2
}
obj.foo1.call(obj2)
obj.foo2.call(obj2)
function foo() {console.log(this) // Windowvar test = () => {console.log(this) // Window}return test
}
var obj = {a: 1,foo: foo
}
var obj2 = {a: 2,foo: foo
}
foo().call(obj)
const obj = {// 普通函数:this指向调用它的对象fn1: function() {console.log(this) // {fn1: ƒ, fn2: ƒ, fn3: ƒ}},// 箭头函数:this指向是父级程序的this指向// 父级程序是obj对象,但只有函数有this,obj对象没有this// 父级程序没有this,指向的是windowfn2: () => {console.log(this) // Window对象}, // fn3是一个普通函数:this指向的是obj对象fn3: function() {// fn4是一个箭头函数:this指向的是父级程序的this指向// 父级程序是fn3,fn3的this指向的是obj对象,所以fn4箭头函数的this也是指向obj对象const fn4 = () => {console.log(this) // {fn1: ƒ, fn2: ƒ, fn3: ƒ}}fn4()}
}
obj.fn1()
obj.fn2()
obj.fn3()
var x = 11;
var obj = {x: 22,y: this,say: () => {console.log(this.x);}
}
obj.say();					// 11
console.log(obj.y);	// window

解析:obj 对象中的 this 指向的就是 window,也就是全局环境。所以obj.y打印的是window对象;**因为箭头函数中的 this 指向父级程序的指向。**从以上例子可以看出来,父级obj指向了window,所以this.x打印的是11

var a = 11;
function fn() {this.a = 22;let b = () => {console.log(this.a)}b();
}
var x = new fn();	// 22

解析:箭头函数中会往上寻找this,直到找到所代表的this为止。例子中,构造函数被实例化成为一个对象x,那x中的this指代的就是对象x本身,所以箭头函数this就代表x对象,x对象内部的a值为22,所以输出22。

var name = 'window'
var obj1 = {name: '1',fn1: function(){ console.log(this.name) },fn2: () => console.log(this.name),fn3: function (){return function(){console.log(this.name)}},fn4: function(){return () => console.log(this.name)}
}
var obj2 = {name: '2'
};obj1.fn1();                 // 1
obj1.fn1.call(obj2);        // 2
obj1.fn2();                 // window
obj1.fn2.call(obj2);        // window
obj1.fn3();                 // window
obj1.fn3().call(obj2);      // 2
obj1.fn3.call(obj2)();      // window
obj1.fn4();                 // 1
obj1.fn4().call(obj2);      // 1
obj1.fn4.call(obj2)();      // 2
function Foo() {getName = function () {alert(1);};return this;
}
Foo.getName = function () {alert(2);
};
Foo.prototype.getName = function () {alert(3);
};
var getName = function () {alert(4);
};
function getName() {alert(5);
}Foo.getName();						// 2
getName();								// 4
Foo().getName();					// 1
getName();								// 1new Foo.getName();				// 2
new Foo().getName();			// 3
new new Foo().getName();	// 3

解析:

预编译中声明提升:function getName(){} 属于函数声明式,提升到最前面。

所以,全局的getName被替换成为function(){ alert(4) }

执行完Foo()后,函数返回了this,this指向window。所以Foo().getName()相当于window.getName(),而函数中的getName是全局的,执行函数的时候,替换掉了之前的输出为4的getName,当前就是输出1

9、资料

  • 【前端面经】JS this 基本指向原则解析

  • 普通函数和箭头函数this 指向的区别?如何改变this指向?

http://www.lryc.cn/news/486285.html

相关文章:

  • DB Type
  • python-返回函数
  • python语言基础-5 进阶语法-5.2 装饰器-5.2.1 闭包
  • 用vscode编写verilog时,如何有信号定义提示、信号定义跳转(go to definition)、模块跳转(跨文件跳转)这些功能
  • MQTT+Springboot整合
  • ERROR TypeError: AutoImport is not a function
  • 软考教材重点内容 信息安全工程师 第 3 章 密码学基本理论
  • 微信小程序 https://thirdwx.qlogo.cn 不在以下 downloadFile 合法域名列表中
  • Linux性能优化之火焰图的起源
  • 《Markdown语法入门》
  • Controller Baseband commands速览
  • Redisson 3.39.0 发布
  • 高阶C语言补充:柔性数组
  • S32K324信息安全-使用IC5000/IC5700进行debug口解锁
  • 简单实现QT对象的[json]序列化与反序列化
  • Unity肢体控制(关节控制)
  • Node.js | Yarn下载安装与环境配置
  • WPF如何全局应用黑白主题效果
  • [Qt] Qt删除文本文件中的某一行
  • 【HarmonyOS学习日志(9)】一次开发,多端部署之界面级一多开发
  • 基于Java+SSM+JSP+MYSQL实现的宠物领养收养管理系统功能设计与实现六
  • Java项目实战II基于微信小程序的课堂助手(开发文档+数据库+源码)
  • 解析 Android WebChromeClient:提升 WebView 用户体验的关键组件
  • 【LeetCode热题100】字符串
  • OceanBase 闪回查询
  • C++析构函数详解
  • 【网络安全 | 漏洞挖掘】未授权获取AI聊天内容
  • 时间序列分析——移动平均法、指数平滑法、逐步回归法、趋势外推法等(基于Python实现)
  • opencv(c++)----图像的读取以及显示
  • PyTorch——从入门到精通:PyTorch基础知识(张量)【PyTorch系统学习】