JavaScript:this(一)
本文和大家聊聊this
,this
在JavaScript
中可说是神一样的存在,灵活性太强了,早期对this
做过一次梳理,但现在面对this
的使用还是怕怕的,对this
的理解不透彻,还是在猜结果,根据应用场景划分梳理,更多的是在硬记,面对复杂恶心人的应用场景,特别是在面试的时候出的题,还是不明白,对结果一头雾水。本文,努力从底层和规范梳理this
,真正的理解this
。
前面的文章有写到,程序执行调用前会创建对应的执行上下文,一个执行上下文可以理解为一个抽象的对象,其中this就是这个对象的一个属性:
executionContext: {
variable object:vars, functions, arguments
scope chain: variable object + all parents scopes
thisValue: context object
}
this
值在进入上下文时就已经确定了,并且在上下文运行期间永久不变。也就是说this
是在函数调用的时候确定的。
我们看看ECMAScript
规范(5.1)中怎么描述this
:
The this keyword evaluates to the value of the ThisBinding of the current execution context.
意思就是说this
执行为当前执行环境(执行上下文
)的ThisBinding
。ThisBinding
就是this
的值。
this
也是一个对象,与执行的上下文环境息息相关,也可以把this
称为上下文对象,激活执行上下文的上下文。
在描述this
之前,我们先来看几个比较重要的概念。
类型
我们都知道在JavaScript
中对象是引用类型,函数和数组都属于引用类型,还有基本类型:Number
、String
、Boolean
等。其实在ECMASciript5.1
规范中将类型分为了两种:
Algorithms within this specification manipulate values each of which has an associated type. The possible value types are exactly those defined in this clause. Types are further subclassified into ECMAScript language types and specification types.
规范的算法操作各个有类型的值,可处理的类型在算法相关叙述中定义。类型又再分为ECMAScript
语言类型和规范类型。
An ECMAScript language type corresponds to values that are directly manipulated by an ECMAScript programmer using the ECMAScript language. The ECMAScript language types are Undefined, Null, Boolean, String, Number, and Object.
ECMAScript
语言类型就是ECMAScript
开发者使用ECMAScript
语言直接操作的值对应的类型。ECMAScript
语言类型包括未定义(Undefined
)、空(Null
)、布尔值(Boolean
)、字符串(String
)、数值(Number
)、对象(Object
)。也就是我们常说未定义(Undefined
)、空(Null
)、布尔值(Boolean
)、字符串(String
)、数值(Number
)为基本数据类型,对象(Object
)为引用类型。
A specification type corresponds to meta-values that are used within algorithms to describe the semantics of ECMAScript language constructs and ECMAScript language types. The specification types are Reference, List, Completion, Property Descriptor, Property Identifier, Lexical Environment, and Environment Record. Specification type values are specification artefacts that do not necessarily correspond to any specific entity within an ECMAScript implementation. Specification type values may be used to describe intermediate results of ECMAScript expression evaluation but such values cannot be stored as properties of objects or values of ECMAScript language variables.
规范类型是描述ECMAScript
语言构造与语言类型语意的算法所用的元值对应的类型。规范类型包括引用(Reference
)、列表(List
)、完结(Completion
)、属性描述式(Property Descriptor
)、属性标识(Property Identifier
)、词法环境(Lexical Environment
)、环境记录(Environment Record
)。
我们需要知道规范类型的值是不一定对应ECMAScript
实现里的任何实体的虚拟对象。规范类型可以用来描述ECMAScript
表示运算的中途结果,但这些值不能存成对象的属性或是ECMAScript
语言变量的值。
对于规范类型不理解的话,我们只需要知道,规范类型用来描述表达式求值过程的中间结果,是一种内部实现,是用来描述语言底层行为逻辑,不对开发者直接开放。
尤雨溪大大在知乎中就Reference
的提问也做来解答:

ECMASciript
规范对类型做了很明确的定义,类型分为ECMAScript
语言类型和规范类型,其中规范类型中有讲到引用(Reference
),需要注意的是这里的引用,不同与我们常说的对象(Object
)为引用类型中的引用,这是两个概念,对象(Object
)为引用类型中的引用更多的是强调在我们定义对象的时候,通过声明变量,将值赋值给变量,变量存的是这个对象的内存地址,并不是对象的值本身,这也就是引用传递。而规范类型中提及到的引用(Reference
)并不是这个意思,规范中是怎么定义的,我们下面来看看,理解规范类型中提及到的引用(Reference
)对我们理解this
有非常大的帮助,是重点:
Reference
Reference
(引用),这里说的引用是规范类型中的引用。我们先来看看规范中怎么定义引用规范类型的:
The Reference type is used to explain the behaviour of such operators as delete, typeof, the assignment operators, the super keyword and other language features. For example, the left-hand operand of an assignment is expected to produce a reference.
上面是ECMAScript6
规范中的定义,比ECMASciript5.1
定义更丰富。
Reference
类型是ECMAScript
规范用来解释delete
, typeof
、赋值运算符、super
关键字等语言特性的行为。例如在赋值运算中左边的操作数期望产生一个引用。
Reference
类型值可以理解为是对某个变量、数组元素或者对象属性所在的内存地址的引用,不是对其所在内存地址所保存值的引用。在 ECMAScript
中,赋值运算符的左侧是一个引用(Reference
),不是值。
我们再来看看Reference
的具体内容:
A Reference is a resolved name binding. A Reference consists of three components, the base value, the referenced name and the Boolean valued strict reference flag. The base value is either undefined, an Object, a Boolean, a String, a Number, or an environment record (10.2.1). A base value of undefined indicates that the reference could not be resolved to a binding. The referenced name is a String.
一个引用(Reference
)是一个已解决的命名绑定。一个引用(Reference
)由三部分组成,基(base
)值,引用名称(referenced name
)和布尔值、严格引用(strict reference
)标志。基值是 undefined
, 一个Object
, 一个Boolean
, 一个String
, 一个Number
, 一个environment record
中的任意一个。基值是undefined
表示此引用可以不解决一个绑定。引用名称是一个字符串。
我们来看一个简单的实例,理解下引用(Reference
):
// 实例一
var a = 1;
上面就是一个很简单的声明变量,变量a
对应的Reference
是什么呢:
// var a = 1 对应的Reference
var aReference = {
base: Environment Record,
name: 'a',
strict: false
}
有人会觉得根据定义好像是这么写的,也有人会思考为什么是这么写:
这是在全局环境下声明了一个变量,并且做了简单的赋值操作,我们来看看规范中怎么定义简单赋值的:
11.13.1 Simple Assignment ( = )
The production AssignmentExpression : LeftHandSideExpression = AssignmentExpression is evaluated as follows:
- Let lref be the result of evaluating LeftHandSideExpression.
- Let rref be the result of evaluating AssignmentExpression. ...
- Return rval.
我们只要看第一步就可以了,令lref
解释执行LeftHandSideExpression
的结果。
LeftHandSideExpression
也就是声明的变量a
,我们知道声明的变量a
可以称为是一个标识符(Identifier
),为了帮助大家更好的理解,我们来看看LeftHandSideExpression
中有没有标识符,先看下LeftHandSideExpression
是怎么定义的:
LeftHandSideExpression :
NewExpression CallExpression
规范中写明了LeftHandSideExpression
包含了两种,一是new
表达式,二是函数调用表达式,并没有我们想要看到的标识符。我们再追根下,new
表达式和函数调用表达式又包含了哪些内容:
NewExpression :
MemberExpression new NewExpression
CallExpression :
MemberExpression Arguments CallExpression Arguments CallExpression [ Expression ] CallExpression . IdentifierName
上面的new
表达式和函数调用表达式同样也没看到我们想要看到的标识符,我们就再追根下,我们看到new表达式中可以是成员表达式(MemberExpression
),我们再来看看成员表达式中有没有我们想要的内容:
MemberExpression :
PrimaryExpression FunctionExpression MemberExpression [ Expression ] MemberExpression . IdentifierName new MemberExpression Arguments
在上面的内容中还是没有看到我们想要看到的标识符,怎么办,只有在追下了,我们再看看PrimaryExpression
中有什么:
PrimaryExpression :
this Identifier Literal ArrayLiteral ObjectLiteral ( Expression )
这下在PrimaryExpression
看到了我们想要看到的标识符(Identifier
)了,所以确保变量a
是标识符无疑了,为了确定它也挺不容易的。
既然我们现在知道了变量a
是一个标识符,我们再来看看标识符是怎么执行的:
11.1.2 Identifier Reference
An Identifier is evaluated by performing Identifier Resolution as specified in 10.3.1. The result of evaluating an Identifier is always a value of type Reference.
Identifier
的执行遵循10.3.1
所规定的标识符查找。标识符执行的结果总是一个Reference
类型的值。也就说明了a
这个标识符是一个Reference
类型的值。我们再来看看10.3.1
写了什么:
10.3.1 Identifier Resolution
Identifier resolution is the process of determining the binding of an Identifier using the LexicalEnvironment of the running execution context. During execution of ECMAScript code, the syntactic production PrimaryExpression : Identifier is evaluated using the following algorithm:
- Let env be the running execution context’s LexicalEnvironment.
- If the syntactic production that is being evaluated is contained in a strict mode code, then let strict be true, else let strict be false.
- Return the result of calling GetIdentifierReference function passing env, Identifier, and strict as arguments.
The result of evaluating an identifier is always a value of type Reference with its referenced name component equal to the Identifier String.
标识符解析是指使用正在运行的执行环境中的词法环境,遇到一个标识符获得其对应的绑定过程。在ECMAScript
代码执行过程中,PrimaryExpression : Identifier
这一语法产生式将按以下算法进行解释执行:
- 令
env
为正在运行的执行环境的词法环境。 - 如果正在解释执行的语法产生式处在严格模式下中的代码,则令
stric
t的值为true
,否则令strict
的值为false
。 - 以
env
,Identifier
和strict
为参数,调用GetIdentifierReference
函数,并返回调用的结果。
解释执行一个标识符得到的结果必定是引用类型的对象,且其引用名属性的值与Identifier
字符串相等。
上面提到了GetIdentifierReference
函数,我们再来看看这是什么:
10.2.2.1 GetIdentifierReference (lex, name, strict) .... Return a value of type Reference whose base value is envRec, whose referenced name is name, and whose strict mode flag is strict. ....
规范中写到返回一个类型为 引用(Reference
) 的对象,其基(base
)值为envRec
,引用的名称(name
)为 name
,严格模式标识的值为strict
就本实例来说,base
值为envRec
,envRec
也就是10.3.
1传入的执行环境的词法环境。而一个词法环境是由一个环境记录项(Environment Record
)和可能为空的外部词法环境引用构成。我们暂时不用管外部词法环境,我们需要知道环境记录项这个概念。
在规范中,有两种环境记录项,一是声明式环境记录项,另一种是对象式环境记录项。声明式环境记录项定义那些将标识符与语言值直接绑定的ECMA
脚本语法元素,例如函数定义,变量定义以及Catch
语句。所以本实例的环境记录项是声明式环境记录项。
说了这么多就是为了告诉大家本实例基(base
)值是Environment Record
,name
是foo
,所以抽象数据结构为:
var aReference = {
base: Environment Record,
name: 'a',
strict: false
}
规范中也使用了以下抽象操作来接近引用:
GetBase(V)
。 返回引用值V
的基值组件。GetReferencedName(V)
。 返回引用值V
的引用名称组件。IsStrictReference(V)
。 返回引用值V
的严格引用组件。HasPrimitiveBase(V)
。 如果基值是Boolean
,String
,Number
,那么返回true
。IsPropertyReference(V)
。 如果基值是个对象或HasPrimitiveBase(V)
是true
,那么返回true
;否则返回false
。IsUnresolvableReference(V)
。 如果基值是undefined
那么返回true
,否则返回false
。
规范使用以下抽象操作来操作引用:
GetValue(v)
。返回对象属性真正的值,是reference
传入,会返回一个普通类型出来。比如foo
为reference
,通过GetValue
之后就是一个普通的object
,也就是foo
对应的js
类型本身PutValue(v,w)
。
铺垫了这么多,现在我们来聊聊this
了,有上面的理论基础理解this
相对会简单很多:
this
绑定多数是出现在函数调用的时候,不同的场景下,this
的指向不一样,我们先来看看函数调用在规范中怎么定义的:
11.2.3 Function Calls
The production CallExpression : MemberExpression Arguments is evaluated as follows:
- Let ref be the result of evaluating MemberExpression.
- Let func be GetValue(ref).
- Let argList be the result of evaluating Arguments, producing an internal list of argument values (see 11.2.4).
- If Type(func) is not Object, throw a TypeError exception.
- If IsCallable(func) is false, throw a TypeError exception.
- If Type(ref) is Reference, then
a. If IsPropertyReference(ref) is true, then
i. Let thisValue be GetBase(ref).
b. Else, the base of ref is an Environment Record
i. Let thisValue be the result of calling the ImplicitThisValue concrete method of GetBase(ref).
- Else, Type(ref) is not Reference.
a. Let thisValue be undefined.
- Return the result of calling the [[Call]] internal method on func, providing thisValue as the this value and providing the list argList as the argument values.
函数调用具体内容如下:
- 令
ref
为解释执行MemberExpression
的结果 - 令
func
为GetValue(ref)
- 令
argList
为解释执行Arguments
的结果 , 产生参数值们的内部列表(see 11.2.4
) - 如果
Type(func)
is notObject
,抛出一个TypeError
异常 - 如果
IsCallable(func)
isfalse
,抛出一个TypeError
异常 - 如果
Type(ref)
为Reference
,
- 如果
IsPropertyReference(ref)
为true
thisValue
为GetBase(ref)
ref
的基值是一个环境记录项thisValue
为调用GetBase(ref)
的ImplicitThisValue
具体方法的结果
Type(ref)
不是Reference
- 令
thisValue
为undefined
- 返回调用
func
的[[Call]]
内置方法的结果 , 传入thisValue
作为this
值和列表argList
作为参数列表
规范中讲到了函数调用时如何确定this
的值,我们下面就按照上述步骤来解析不同场景下this
值是什么。
默认绑定
我们先看一个最简单的场景:
// 实例二
function foo() {
console.log(this); // window
}
foo();
函数foo
执行输出想必大家都知道输出是window
,我们从规范中来分析下为什么this
指向的是window
:
我们现在只需要看第一步,令ref
为解释执行MemberExpression
的结果。MemberExpression
是什么呢?我们来看看规范是怎么定义的:
11.2 Left-Hand-Side Expressions
MemberExpression :
PrimaryExpression // 主值表达式 FunctionExpression // 函数调用表达式 MemberExpression[Expression] // 属性访问表达式 MemberExpression.IdentifierName // 属性访问表达式 new MemberExpression Arguments // 对象创建表达式
MemberExpression
可以理解为()左边部分,在这里是foo
,foo
是主值表达式(PrimaryExpression
)中的一种,我们先看下PrimaryExpression
是怎么定义的:
11.1 Primary Expressions
PrimaryExpression :
this Identifier Literal ArrayLiteral ObjectLiteral ( Expression )
看的出来foo
就是一个标识符(Identifier
),前面的内容也已经讲过了,解释执行一个标识符得到的结果必定是引用类型的对象,且其引用名属性的值与Identifier
字符串相等。
所以foo
的抽象数据结构如下:
foo_reference = {
base: Environment Record,
name: "foo",
strict: false
}
到这里第一步已经完成了,执行MemberExpression
的结果就是foo_reference
,ref = foo_reference
,ref
是一个Reference
。
因为ref
是Reference
,所以来到了第六步:如果Type(ref)
是Reference
,那么如果IsPropertyReference(ref)
为true
,那么令thisValue
为GetBase(ref)
. 否则, ref
的基值是一个环境记录项,令 thisValue
为调用GetBase(ref)
的ImplicitThisValue
具体方法的结果。
ref
的基(base
)值是一个环境记录项(Environment Record
),不是一个对象,所以IsPropertyReference(ref)
是false
,this
的值是调用GetBase(ref)
的ImplicitThisValue
具体方法的结果。
我们来看看ImplicitThisValue
方法是什么:
10.2.1.1.6 ImplicitThisValue()
Declarative Environment Records always return undefined as their ImplicitThisValue.
声明式环境记录项永远将undefined
作为其ImplicitThisValue
返回.
函数foo
执行到这里,thisValue = undefined
,对应到JavaScript
代码中的this
,this=undefined
,还没有结束,规范中在进入函数代码有写到:
10.4.3 Entering Function Code
Else if thisArg is null or undefined, set the ThisBinding to the global object.
如果thisArg
是null
或undefined
,则设this
绑定为 全局对象 。所以函数foo
执行时,this
指向全局对象,也就是window
。
// 实例二
function foo() {
console.log(this); // window
}
foo();
所以这个实例控制台输出的就是window
。这也就是从应用场景来区分的话,就是默认绑定。
这就是this
指向解析的全过程了,从规范中解读this
的指向,从底层了解this
的指向,不需要去区分默认绑定、隐性绑定、显性绑定等这种从应用场景来记this
的指向,记不住也容易出错,遇到复杂的场景还是容易犯错,我们需要真正的从底层来理解this
,对于this
理解透彻。
有了上面的分析方式,我们再看看几种比较常见的this
应用场景:
隐性绑定
// 实例三
var obj = {
foo: function() {
console.log(this); // obj
}
}
obj.foo();
实例三中MemberExpression
计算结果是obj.foo
,obj.foo
是个什么,会是不是一个Reference
呢?
按照前面的分析方式,我们知道obj.foo
是MemberExpression.IdentifierName
,是一个属性访问。
我们再来看看规范中怎么定义属性访问(Property Accessors
)的:
11.2.1 Property Accessors
The production MemberExpression : MemberExpression[Expression] is evaluated as follows:
- Let baseReference be the result of evaluating MemberExpression
- Let baseValue be GetValue(baseReference).
- Let propertyNameReference be the result of evaluating Expression.
- Let propertyNameValue be GetValue(propertyNameReference).
- Call CheckObjectCoercible(baseValue).
- Let propertyNameString be ToString(propertyNameValue).
- If the syntactic production that is being evaluated is contained in strict mode code, let strict be true, else let strict be false.
- Return a value of type Reference whose base value is baseValue and whose referenced name is propertyNameString, and whose strict mode flag is strict.
上面主要讲的是:
- 令
baseReference
为解释执行MemberExpression
的结果 - 令
baseValue
为GetValue(baseReference)
- 令
propertyNameReference
为解释执行Expression
的结果 - 令
propertyNameValue
为GetValue(propertyNameReference)
- 调用
CheckObjectCoercible(baseValue)
- 令
propertyNameString
为ToString(propertyNameValue)
- 如果正在执行中的语法产生式包含在严格模式代码当中,令
strict
为true
, 否则令strict
为false
- 返回一个值类型的引用,其基值为
baseValue
且其引用名为propertyNameString
, 严格模式标记为strict
我们回到我们的实例三,obj.foo
中的MemberExpression
是obj
,执行MemberExpression
的结果返回一个Reference
,假设是reference_obj
,baseReference
为reference_obj
。
baseValue
就是GetValue(reference_obj)
,GetValue
方法上面也有介绍,baseValue
也就是obj
。
同理propertyNameString
就是foo
再看最后一步(第八步),所以obj.foo
最终执行是一个Reference
,且数据结构如下:
reference_obj_foo = {
base: obj,
name: "foo",
strict: false
}
因为obj.foo一个Reference
,ref = reference_obj_foo
,来到了函数调用的第六步:
If Type(ref) is Reference, then
a. If IsPropertyReference(ref) is true, then
i. Let thisValue be GetBase(ref).
IsPropertyReference()
方法前面也有介绍,如果基值是个对象或HasPrimitiveBase(V)
是true
,那么返回true
。本实例base
是obj
,是一个对象,所以IsPropertyReference(ref)
返回true
,也因为返回了true
,所以thisValue = GetBase(ref)
,GetBase(V)
方法前面也有介绍,返回引用值V
的基值组件,即thisValue = GetBase(ref) = obj
。
所以obj.bar()
执行时this
指向obj
,这也就是this
的隐性绑定。
现在已经分析了this
的基本指向,默认绑定和隐性绑定的指向问题,按照应用场景来分的话,还有显性绑定、new
绑定以及箭头函数等,由于内容还有很多下篇文章再详细输出,本文就暂时写到这。本文也基本的讲述从规范的角度this
的指向,从规范中帮助我们更好的理解this
。
结语
文章如有不正确的地方欢迎各位大佬指正,也希望有幸看到文章的同学也有收获,一起成长!
-------------------------------本文首发于个人公众号------------------------------

转载自:https://juejin.cn/post/6844903936374095886