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

一个没有手动加分号引发的bug

最近因为分号的疏忽,导致出现了一个bug,记录下来,分享给大家。

1、一个示例

给你下面这一段代码,你根据经验判断一下运营结果

let [a,b] = ['a','b']
let [x,y] = [1,2]
if(x < y){[x,y] = [y,x][a,b] = [b,a]
}

按照一般的理解,是不是应该是 x = 2,y=1,a = ‘b’,b = ‘a’ ?
可实际呢?咱们跑一下看看:

console.log([x,y]) // ['b', 'a']
console.log([a,b]) // ['a','b']

2、为什么呢?

这段代码不加分号会导致 [x,y][a,b] 被错误解析为一个连续表达式,从而引发赋值错误。根本原因是 JavaScript 的 自动分号插入(ASI) 机制在以下情况 不会插入分号

  1. 当下一行以 [ 开头时,会被解析为当前语句的延续
  2. 赋值表达式可以跨行解析

3、错误解析过程(无分号时):

if(x < y){[x,y] = [y,x]  // 注意这里没有分号[a,b] = [b,a]  // 被解析为上一行的延续
}

JavaScript 引擎会将其解析为:

[x,y] = [y,x][a,b] = [b,a]

这实际等价于:

// 1. 先计算 [y,x][a,b]
// - [y,x] 是一个数组 [2, 1] 
// - [a,b] 是逗号表达式,返回最后一个值 'b'
const temp = [y,x]['b'] // 相当于访问数组的 'b' 属性
// 2. 将 [b,a] 赋值给上述结果
temp = [b,a]
// 再将 temp 赋值给 [x,y] 
[x,y] = temp
// 也就是
[x,y] = [b,a]

最终导致 xy 被赋值为 [b,a](即 ['b','a']

4、加分号后验证代码:

let [a,b] = ['a','b'];  // a='a', b='b'
let [x,y] = [1,2];      // x=1, y=2if(x < y){[x,y] = [y,x];      // 正常交换:x=2, y=1[a,b] = [b,a];      // 正常交换:a='b', b='a'
}console.log(x, y); // 输出: 2 1
console.log(a, b); // 输出: b a

5、解决方案:

5.1. 始终添加分号(推荐)
// 正确写法
[x,y] = [y,x]; // 明确分号结束
[a,b] = [b,a];
5.2. 使用防御性分号
// 在可能引起歧义的语句前加分号
;[a,b] = [b,a]
5.3. 用逗号写成单行
// 单行写法避免换行问题
if(x < y){ [x,y] = [y,x], [a,b] = [b,a] }

6、关键教训:

  1. 避免以 [( 开头的行
    这类语法结构容易与前一行的表达式粘连

  2. 解构赋值后必须加分号
    特别是在块语句(if/for 等)内部

  3. 使用 ESLint 规则
    配置 semi: ["error", "always"] 强制分号使用:

    // .eslintrc.json
    {"rules": {"semi": ["error", "always"]}
    }
    
http://www.dtcms.com/a/294016.html

相关文章:

  • python---元组(Tuple)
  • C#简介(从入门到精通)
  • 判断矩形能否放入多边形内——cad c# 二次开发实现
  • 【服务器】 MCTP Over PCIe 的内容、用途、工作原理及硬件设计注意事项
  • 数据结构(2)顺序表算法题
  • C#使用socket报错 System.Net.Sockets.SocketException:“在其上下文中,该请求的地址无效。
  • .net平台的跨平台桌面应用开发的技术方案总结对比
  • 【黑马SpringCloud微服务开发与实战】(六)分布式事务
  • Matlab学习笔记:逻辑基础
  • PyTorch武侠演义 第一卷:初入江湖 第6章:驿站的秘密信鸽
  • Apache JMeter 使用记录踩坑
  • 前端模块化:CommonJS 与 ES Module
  • 性能测试-jmeter实战5
  • 4️⃣字典(dict)速查表
  • I2C控制器
  • 传统RNN模型笔记:输入数据长度变化的结构解析
  • 通用图片 OCR 到 Word API 数据接口
  • 数据结构自学Day13 -- 快速排序--“前后指针法”
  • 显微科研中的关键选择:不同显微镜相机技术特性与应用适配性全面解析
  • SpringCloudGateWay 使用nacos网关自动负载均衡
  • nrm指南
  • Sklearn 机器学习 线性回归
  • 解决sparksql创建出来的数据库路径错误的问题
  • Docker Desktop 打包Unity WebGL 程序,在Docker 中运行Unity WebGL 程序
  • Unity国际版下载链接分享(非c1国内版)
  • Java面试题035:一文深入了解Docker
  • SQL基础⑧ | 表格篇
  • Android插件化实现方案深度分析
  • 智能小e-外联系统文档
  • CentOS 7 Linux 用 yum 安装 Docker,含 Docker 镜像无法拉取问题(即 docker pull 失败)的解决方案