理解JavaScript中的LHS和RHS

参考资料:凯尔·辛普森《你不知道的JavaScript(上卷)》

1. 编译原理

JavaScript引擎首先会在代码执行前对其进行编译

var a = 2

这行代码将被拆分成两个步骤:

  1. var a
    编译器会在当前作用域中声明一个变量(如果之前没有声明过)
  2. a = 2
    运行时引擎会在作用域中查找该变量,如果能够找到就会对它赋值。

在查找变量的过程中,分为LHS和RHS两种查找方式。

2. LHS和RHS的概念

LHS和RHS的字面意思是Left Hand Side和Right Hand Side,于是有一种理解是把其当作运算符的左侧和右侧。
按照《你不知道的JavaScript》的说法是:

讲得更准确一点,RHS查询与简单地查找某个变量的值别无二致,而LHS查询则是试图找到变量的容器本身,从而可以对其赋值。从这个角度说,RHS并不是真正意义上的“赋值操作的右侧”,更准确地说是“非左侧”。你可以将RHS理解成retrieve his source value(取到它的源值),这意味着“得到某某的值”。

所以,可以这样理解:
LHS:赋值操作的目标是谁,如:

a = 2

这里对a的引用则是LHS引用,因为实际上我们并不关心当前的值是什么,只是想要为=2这个赋值操作找到一个目标。

RHS:赋值操作的源头是谁,如

console.log(a)

因为这里a并没有赋予任何值。相应地,需要查找并取得a的值,这样才能将值传递给console.log(..)

3. 二者结合的例子

function foo(a) {
    console.log(a)
}
foo(2)

找出其中所有的RHS查询:

  1. foo() 需要查找到foo方法
  2. log() 需要查找到console下的log方法

找出其中所有的LHS查询:

  1. a = 2 这里存在隐式变量赋值,执行函数时,将2通过参数的形式赋值给a
  2. 假设在log(..)函数的原生实现中它可以接受参数,在将2赋值给其中第一个(也许叫作arg1)参数之前,这个参数需要进行LHS引用查询。

再看另一个例子:

function foo(a) {
    var b = a;
    return a + b;
}
var c = foo(2);
  1. 找出其中所有的LHS查询

    1. c = ... 赋值什么给c?
    2. a = 2 隐式变量分配
    3. b = ... 赋值什么给b?
  2. 找出其中所有的RHS查询

    1. foo(...) 执行谁?
    2. = a a赋值给谁?
    3. a... a和谁进行运算?
    4. ...b 谁和b进行运算?
4. 遍历规则

LHS和RHS查询都会在当前执行作用域中开始,如果有需要(也就是说它们没有找到所需的标识符),就会向上级作用域继续查找目标标识符,这样每次上升一级作用域(一层楼),最后抵达全局作用域(顶层),无论找到或没找到都将停止。

5. 异常
function foo(a) {
    console.log(a+b)
    b = a
}

foo(2)

如果RHS查询在所有嵌套的作用域中遍寻不到所需的变量,引擎就会抛出ReferenceError异常。
如果RHS查询找到了一个变量,但是你尝试对这个变量的值进行不合理的操作,比如试图对一个非函数类型的值进行函数调用,或者引用nullundefined类型的值中的属性,那么引擎会抛出另外一种类型的异常,叫作TypeError

不成功的LHS引用会导致自动隐式地创建一个全局变量(非严格模式下),该变量使用LHS引用的目标作为标识符,或者抛出ReferenceError异常(严格模式下)。

ReferenceError同作用域判别失败相关。
TypeError代表作用域判别成功了,但是对结果的操作是非法或不合理的。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值