写了10年的Javascript也未必全了解的连续赋值运算


Posted in Javascript onMarch 25, 2011

一、引子

var a = {n:1}; 
a.x = a = {n:2}; 
alert(a.x); // --> undefined

这是蔡蔡在看 jQuery源码 时发现这种写法的。以上第二句 a.x = a = {n:2} 是一个连续赋值表达式。这个连续赋值表达式在引擎内部究竟发生了什么?是如何解释的?

二、猜想

猜想1:从左到右赋值,a.x 先赋值为{n:2},但随后 a 赋值为 {n:2},即 a 被重写了,值为 {n:2},新的 a 没有 x属性,因此为undefined。步骤如下
1, a.x = {n:2};
2, a = {n:2};
这种解释得出的结果与实际运行结果一致,貌似是对的。注意猜想1中 a.x 被赋值过。
猜想2:从右到左赋值,a 先赋值为{n:2},a.x 发现 a 被重写后(之前a是{a:1}),a.x = {n:2} 引擎限制a.x赋值,忽略了。步骤如下:
1, a = {n:2};
2, a.x 未被赋值{n:2}
等价于 a.x = (a = {n:2}),即执行了第一步,这样也能解释a.x为undefined了。注意猜想2中a.x压根没被赋值过。

三、证明

上面两种猜想相信多数人都有,群里讨论呆呆认为是猜想1, 我认为是猜想2。其实都错了。我忽略了引用的关系。如下,加一个变量b,指向a。

var a = {n:1}; 
var b = a; // 暂存a 
a.x = a = {n:2}; 
alert(a.x);// --> undefined 
alert(b.x);// --> [object Object]

发现a.x仍然是undefined,神奇的是 b.x 并未被赋值过(比如:b.x={n:2}),却变成了[object Object]。b 是指向 a({n:1})的,只有a.x = {n:2}执行了才说明b是有x属性的。实际执行过程:从右到左,a 先被赋值为{n:2},随后a.x被赋值{n:2}。
1, a = {n:2};
2, a.x = {n:2};
等价于
a.x = (a = {n:2});
与猜想2的区别在于a.x 被赋值了,猜想2中并未赋值。最重要的区别,第一步 a = {n:2} 的 a 指向的是新的对象{n:2} , 第二步 a.x = {n:2} 中的 a 是 {a:1}。即在这个连等语句
a.x = a = {n:2};

a.x 中的a指向的是 {n:1},a 指向的是 {n:2}。如下图

写了10年的Javascript也未必全了解的连续赋值运算
四:解惑

这篇写完,或许部分人看完还是晕晕的。因为里面的文字描述实在是绕口。最初我在理解这个连等赋值语句时

var a = {n:1}; 
a.x = a = {n:2};

认为引擎会限制a.x的重写(a被重写后),实际却不是这样的。指向的对象已经不同了。引擎也没有限制a.x={n:2}的重写。
谢谢所有参与讨论的人:蔡蔡、呆呆、儒儒。这个问题最早是蔡蔡提出的。儒儒在 菜鸟灰呀灰 群里每次的讨论都那么投入,认真,哪怕是别人提出的话题。

五:结束

呵,以另一个连续赋值题结束。fun执行后,这里的 变量 b 溢出到fun外成为了全局变量。想到了吗?

function fun(){ 
var a = b = 5; 
} 
fun(); 
alert(typeof a); // --> undefined 
alert(typeof b); // --> number
Javascript 相关文章推荐
js中top/parent/frame概述及案例应用
Feb 06 Javascript
addEventListener()第三个参数useCapture (Boolean)详细解析
Nov 07 Javascript
javascript中打印当前的时间实现思路及代码
Dec 18 Javascript
JavaScript的作用域和块级作用域概念理解
Sep 21 Javascript
javascript学习指南之回调问题
Apr 23 Javascript
Move.js入门
Feb 08 Javascript
基于angular2 的 http服务封装的实例代码
Jun 29 Javascript
3种vue路由传参的基本模式
Feb 22 Javascript
JS实现读取xml内容并输出到div中的方法示例
Apr 19 Javascript
React styled-components设置组件属性的方法
Aug 07 Javascript
vue-cli3.0使用及部分配置详解
Aug 29 Javascript
Vue中的Props(不可变状态)
Sep 29 Javascript
js 实现图片预加载(js操作 Image对象属性complete ,事件onload 异步加载图片)
Mar 25 #Javascript
基于jquery的3d效果实现代码
Mar 23 #Javascript
jquery 操作表格实现代码(多种操作打包)
Mar 20 #Javascript
jQuery实现的Email中的收件人效果(按del键删除)
Mar 20 #Javascript
jquery图片上下tab切换效果
Mar 18 #Javascript
javascript一些实用技巧小结
Mar 18 #Javascript
jquery获取下拉列表的值为null的解决方法
Mar 18 #Javascript
You might like
如何开始收听短波广播
2021/03/01 无线电
php文件上传表单摘自drupal的代码
2011/02/15 PHP
PHP系列学习之日期函数使用介绍
2012/08/18 PHP
PHP页面实现定时跳转的方法
2014/10/31 PHP
php导出生成word的方法
2015/12/25 PHP
AJAX使用了UpdatePanel后无法使用alert弹出脚本
2010/04/02 Javascript
jQuery中获取Radio元素值的方法
2013/07/02 Javascript
JS获取下拉列表所选中的TEXT和Value的实现代码
2014/01/11 Javascript
node.js中的fs.writeFileSync方法使用说明
2014/12/14 Javascript
JavaScript之Object类型介绍
2015/04/01 Javascript
浅析jQuery操作select控件的取值和设值
2016/12/07 Javascript
Bootstrap CSS组件之大屏幕展播
2016/12/17 Javascript
JS前端开发判断是否是手机端并跳转操作(小结)
2017/02/05 Javascript
JavaScript实现前端分页控件
2017/04/19 Javascript
jQuery实现 RadioButton做必选校验功能
2017/06/15 jQuery
JS鼠标滚动分页效果示例
2017/07/05 Javascript
Node.js中使用mongoose操作mongodb数据库的方法
2017/09/12 Javascript
微信小程序使用二次贝塞尔曲线画波浪
2018/12/25 Javascript
深入探讨JavaScript的最基本部分之执行上下文
2019/02/12 Javascript
Js生成随机数/随机字符串的方法小结【5种方法】
2020/05/27 Javascript
python实现统计代码行数的方法
2015/05/22 Python
在Django的视图中使用数据库查询的方法
2015/07/16 Python
Python实现的爬虫功能代码
2017/06/24 Python
Python wxPython库Core组件BoxSizer用法示例
2018/09/03 Python
python实现Dijkstra算法的最短路径问题
2019/06/21 Python
Django项目中使用JWT的实现代码
2019/11/04 Python
基于PyQT实现区分左键双击和单击
2020/05/19 Python
python 密码学示例——理解哈希(Hash)算法
2020/09/21 Python
商业活动邀请函
2014/02/04 职场文书
保健品市场营销方案
2014/03/31 职场文书
关于爱国的标语
2014/06/24 职场文书
大学拉赞助协议书范文
2014/09/26 职场文书
党员查摆问题及整改措施
2014/10/10 职场文书
法学专业求职信范文
2015/03/19 职场文书
离婚起诉书范文2015
2015/05/19 职场文书
浅谈MySQL之select优化方案
2021/08/07 MySQL