百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 行业知识 > 正文

js中this指向的问题与联系深入探究

citgpt 2024-11-25 10:03 3 浏览 0 评论

1.0 前言

JavaScript 中最大的一个安全问题,也是最令人困惑的一个问题,就是在某些情况下this的值是如何确定的。有js基础的同学面对这个问题基本可以想到:this的指向和函数调用的方式相关。这当然是正确的,然而,这几种方式有什么联系吗?这是我接下来要说明的问题。

js中this指向的问题与联系深入探究

2.0 this从哪里来

this 是js的一个关键字,和arguments类似,它是函数运行时,在函数体内部自动生成的一个对象,只能在函数体内部使用。这句话似乎与认知不同,我们在函数体外部即全局作用域下也能使用this

// 直接在全局作用域下输出this
console.log(this);
// 输出window


但是不要忘记,即便是全局作用域,依旧是运行在window下的,我们写的代码都在window的某个函数中。而这也催生了一种理解this指向的方法:this永远指向调用者(非箭头函数中)。

3.0 作为普通函数调用

函数作为普通函数直接调用(也称为自执行函数)的时候,无论函数在全局还是在另一个函数中,this都是指向window

function fn() {
this.author = 'Wango';
}

fn();
console.log(author);
// Wango

这很好理解,但又不是很好理解,因为在代码中省略了window,补全后就好理解了:this指向的是调用者。


function fn() {
this.author = 'Wango';
}

window.fn();
console.log(window.author);
// Wango


而在内部函数中,自执行函数中的this依旧指向全局作用域,我们
无法通过window.foo()调用函数,但并不妨碍我们先这样理解(具体参见本文最后一部分this的强制转型


function fn() {
function foo() {
console.log(this);
}
foo();
// Window
window.foo();
// TypeError
}

fn();


4.0 作为构造函数调用

在构造函数中,this指向new生成的新对象,即构造函数是通过new调用的,构造函数内部的this当然就应该指向new出来的对象。


function Person(name, age) {
this.name = name;
this.age = age;
console.log(this);
// Person { name: 'Wango', age: 24 }
}

new Person('Wango', 24);

构造函数中的this与构造函数的返回值类型无关,下列代码中p指向了构造函数返回的对象,而不是new出来的对象。当然,这是构造函数的特性,与本主题关系不大。


function Person(name, age) {
console.log(this);
// Person {}
this.name = name;
this.age = age;
console.log(this);
// Person { name: 'Wango', age: 24 }

return {
name: 'Lily',
age: 25
}
}

Person.prototype.sayName = function() {
return this.name + ' ' + this.age
}

const p = new Person('Wango', 24);
console.log(p.sayName());
// TypeError: p.sayName is not a function

5.0 作为对象方法调用

通过对象方法调用时,this指向应该是最明晰的了。与其他面向对象语言的this行为相同,指向该方法的调用者。


function Person(name, age) {
this.name = name;
this.age = age;
}

Person.prototype.sayName = fn;

function fn() {
return this.name + ' ' + this.age
}

const p = new Person('Wango', 24);
console.log(p);
// Person { name: 'Wango', age: 24 }
console.log(p.sayName());
// Wango 24


5.1 通过【】调用对象方法

通常,我们对于对象方法是通过.语法调用,但通过[]也可以调用对象方法,在这种情况下的this指向常常会被我们混淆、忽略。


function fn() {
console.log(this);
}

const arr = [fn, 1];

arr[0]();
// [Function: fn, 1]

function fn2() {
arguments[0]();
}

fn2(fn, 1);
// [Arguments] { '0': [Function: fn], '1': 1 }

在上例中,无论是数组还是伪数组,其本质上都是对象,在通过[]获取函数元素并调用的时候,会改变函数中的this指向,this指向这个数组或伪数组,与对象调用函数的行为一致。

6.0 通过call、apply调用

function fn() {
console.log(this.name);
}

const author = {
name: 'Wango'
}

fn.call(author);
// Wango


这似乎与this永远指向调用者相违背,但一旦我们明白了call函数的实现机制就会明白,这不仅不是违背,反而是佐证。对callapplybind 下面截取call简要说明。


// 保存一个全局变量作为默认值
const root = this;

Function.prototype.myCall = function(context, ...args) {
if (typeof context === 'object') {
// 如果参数是null,使用全局变量
context = context || root;
} else {
// 参数不是对象的创建一个空对象
context = Object.create(null);
}
// 使用Symbol创建唯一值作为函数名
let fn = Symbol();
context[fn] = this;
context[fn](...args);
delete context[fn];
}

call 函数最核心的实现在于context[fn] = this;context[fn](...args);这两行。实际上就是将没有函数调用者的普通函数挂载到指定的对象上,这时this指向与对象调用方法的一致。而delete context[fn];是在调用后立即解除对象与函数之间的关联。

7.0 严格模式下的不同表现

7.1 this强制转型

使用函数的apply()call()方法时,在非严格模式下nullundefined值会被强制转型为全局对象。在严格模式下,则始终以指定值作为函数this的值,无论指定的是什么值。这也是为何在严格模式下,自执行函数的this不再指向window,而是指向undefined的根本原因。

// 定义一个全局变量

color = "red";
function displayColor() {
console.log(this.color);
}
// 在非严格模式下使用call修改this指向,并指定null,或undefined,
displayColor.call(null);
displayColor.call();
// red

// 修改指向无效,传入null或undefined被转换为了window


实际上,我们也可以将自执行函数,如fn(),看作是fn.call()的语法糖,在普通模式下,第一个参数默认为undefined,但被强制转换为window。这也就解释了为何所有自执行函数中this都指向window但无法通过window调用的问题(函数在call函数中挂载到window对象上,执行后被立即删除,所以无法再次通过window访问)。

apply()call()方法在严格模式下传入简单数据类型作为第一个参数时,该简单数据类型会被转换为相应的包装类,而非严格模式不会如此转换。

function foo() {
console.log(this);
}

foo.call(); // Window {}
foo.call(2); // Number {2}


function foo() {
console.log(this);
}

foo.call(); // undefined
foo.call(2); // 2

8.0 箭头函数的this指向

在箭头函数中, this引用的是定义箭头函数的上下文。即箭头函数中的this不会随着函数调用方式的改变而改变。


function Person(name) {
this.name = name;

this.getName = () => console.log(this.name);
}

const p = new Person('Wango');

p.getName();
// Wango

const getName = p.getName;

getName();
// Wango
getName.call({name: 'Lily'});
// Wango

相关推荐

外贸网站建设要多少钱?外贸网站建设周期要多久

大家好,我是【无锡柠萌网络lemon56.com】的小美,今天分享:外贸网站建设要多少钱?外贸网站建设周期要多久您关注的问题·FAQ外贸网站建设中遇到的常见问题,让您少走弯路,提高效率!...

500块搭建一个可以卖货的跨境电商独立站,包括服务器和域名吗?

用500元搭建一个可以卖货的跨境电商独立站,虽然有一定挑战性,但通过精打细算和选择合适的工具和服务,仍然是有可能的。以下是一些建议,帮助你在预算范围内实现这个目标,包括服务器和域名的选择。...

无锡网络公司设计搭建阀门网站一般要多少钱?

无锡网络公司设计搭建阀门网站一般要多少钱?这是一个涉及多方面因素的复杂问题,其费用因网站规模、功能需求、设计复杂度及后续维护等因素而异。首先,从基础成本出发,域名注册是搭建网站的第一步,费用通常在几十...

定制化网站开发大概多少钱

定制化网站开发的价格因项目的复杂性、功能需求、设计要求和开发时间等因素而异。以下是一些常见的定制化网站开发价格范围,供参考:1.简单的定制化网站开发:一般来说,一个简单的定制化网站开发项目可能需要花...

9月安卓手机性能排行榜出炉:前十名差距仅为7%

时间已经进入9月,按照安兔兔目前的安卓手机性能榜单显示,红魔9SPro+依旧力压其他热销手机成为榜首,可见游戏定位的这款手机,在性能上确实非常的出色。不过,笔者也发现了一个问题,其中前十名中得分最高...

装修公司怎么在网络平台接单?实现高效获客

装修行业正经历着深刻的变革,随着消费者需求的日益多样化与个性化,传统的线下获客方式已难以满足装修公司的业务需求。因此,装修公司必须紧跟时代步伐,充分利用网络平台实现高效获客。1、入驻装修接单平台:壹品...

企业如何利用二维码进行线上营销?

#企业如何利用二维码进行线上营销?#企业利用二维码进行线上营销,可以从以下几个方面入手:?1.设计创意二维码?:...

家电销售该怎么线上拓客

社交媒体平台:利用微信、微博、抖音等平台,定期发布家电产品的信息、使用心得、促销活动等内容。引流靠手动一定是不行的,所以一般使用点软件肯定没错,最近用了款比较冷门的APP,"里德助手Plus...

pop社交软件,脱单软件排行榜

在交友软件里面脱单是非常明确的一件事,各年龄各社交平台上的女人都是有所不同的,具体问题具体分析一下。接下来就说说又哪些比较优质的恋爱脱单软件。...

手机处理器最新跑分排行榜,你的手机什么水平?

手机处理器最新跑分排行榜,你的手机什么水平?第1名:天玑9300第2名:骁龙8Gen3第3名:A17Pro第4名:天玑9200第5名:骁龙8Gen2第6名:骁龙8Gen1...

继番茄小说后,字节再推免费网文 App“蛋花小说”和“常读小说”

据Tech星球报道,字节跳动公司近期推出了两款全新的免费网文App,分别是“蛋花小说”和“常读小说”。这两款产品的开发公司分别为湖北福瑞兴网络科技有限公司和湖北聚合润网络科技有限公司,均为字节跳动的1...

全世界最好用的AI软件排名是?

Hey小伙伴们,今天咱们来聊聊那些让人爱不释手的AI神器!在这个智能化时代,谁还没几个拿得出手的AI软件呢?别急,我这就给你盘点一波全球超火的AI软件,保证让你大开眼界!...

手机root软件哪个成功率高?手机root软件排行榜2025

作为游戏玩家,欲在手游中畅玩无阻,root权限不可或缺,可优化画质、启用辅助。2025年已至,究竟哪些手机root软件能脱颖而出?哪款软件的root成功率会独占鳌头?一、手机root软...

独立站新手教程引流篇:如何优化谷歌广告投放效果?

随着谷歌广告单价的持续上涨,如何在提升投放效果的同时,降低推广费用成为每一个独立站卖家的必修课。因此新手卖家在完成初步的广告投放流程后,最重要的就是了解一下谷歌广告优化的基础策略。设置转化跟踪即利用G...

云媒易:干货知识分享!海外推广的渠道有哪些,如何正确的选择?

越来越多的海外企业或国内的跨境商家认识到海外的网络营销的作用,开展线上外贸营销,渠道是非常的重要,那么海外的推广渠道有哪些?企业应该如何选择呢?1、社交媒体推广社交媒体推广是现在海外推广方式中最热门的...

取消回复欢迎 发表评论: