页面白屏了?看看可选链操作符(?.)

开发 前端
可选链操作符(?.)允许读取位于连接对象链深处的属性的值,而不必明确验证链中的每个引用是否有效。

背景

今天又被 QA 找: 这个页面昨天还好好的, 今天就白屏了, 是不是你代码有问题啊, 赶紧看看。

上去一看, 找到了原因:

原本 pickup, dropoff 两个字段没有数据的话, 应该返回{}, 结果现在pickup字段返回了null, 而我们取值的时候,也没对这个地方做防御。

  1. list: openApiOrderInfo.pickup.address_list, 

结果就是:脚本报错, 页面不可用。

解决起来也很简单, 要么给个默认值, 要么使用 ?. 做一层防御。

改完再试一下, 就 OK 了, 页面恢复正常。

下面我们就说一下这个 ?.

今天的主要内容:

  • 什么是可选链操作符(?.)
  • 如何启用这个功能
  • 可选链操作符(?.) 是如何工作的
  • Heny发布的相关些资料
  • 总结

正文语种

可选链操作符(?.), 大家都很熟悉了,这里再简单回顾一下。

什么是可选链操作符(?.)

可选链操作符(?.)允许读取位于连接对象链深处的属性的值,而不必明确验证链中的每个引用是否有效。

比如,思考一个存在嵌套结构的对象 obj。不使用可选链的话,查找一个深度嵌套的子属性时,需要验证之间的引用,例如:

  1. let nestedProp = obj.first && obj.first.second

为了避免报错,在访问obj.first.second之前,要保证 obj.first 的值既不是 null,也不是 undefined。

如果只是直接访问 obj.first.second,而不对 obj.first 进行校验,则有可能抛出错误。

有了可选链操作符(?.),在访问 obj.first.second 之前,不再需要明确地校验 obj.first 的状态,再并用短路计算获取最终结果:

  1. let nestedProp = obj.first?.second

这等价于以下表达式,但实际上没有创建临时变量:

  1. let temp = obj.first
  2. let nestedProp = ((temp === null || temp === undefined) ? undefined : temp.second); 

?. 操作符的功能类似于 . 链式操作符,不同之处在于:

在引用为空(nullish) (null 或者 undefined) 的情况下不会引起错误,该表达式短路返回值是: undefined。

与函数调用一起使用时,如果给定的函数不存在,则返回 undefined。

当尝试访问可能不存在的对象属性时,使用可选链操作符将会使表达式更短、更简明。

有两点需要我们留意:

如果存在一个属性名且不是函数, 使用 ?. 仍然会产生一个 TypeError 异常 (x.y is not a function).

  1. let result = someInterface.customMethod?.(); 

如果 someInterface 自身是 null 或者 undefined ,异常 TypeError 仍会被抛出。

如果你希望允许 someInterface 也为 null 或者 undefined,那么你需要像这样写 someInterface?.customMethod?.()

可选链不能用于赋值

  1. let object = {}; 
  2. object?.property = 1; // Uncaught SyntaxError: Invalid left-hand side in assignment 

如何启用这个功能

  1. // install 
  2. npm install --save-dev @babel/plugin-proposal-optional-chaining 
  3.  
  4. // babel config 
  5.  "plugins": [ 
  6.     "@babel/plugin-proposal-optional-chaining" //可选链 
  7.     "@babel/plugin-proposal-nullish-coalescing-operator", //双问号 
  8.   ] 

可选链操作符(?.) 是如何工作的

  1. const a = { 
  2.   b: 1 
  3. }; 
  4.  
  5. console.log(a?.b); 

会被转换成:

  1. const a = { 
  2.   b: 1 
  3. }; 
  4.  
  5. console.log(a === null ? void 0 : a.b); 

 

如果层级更深, 会创建临时变量用于记录:

  1. const a = { 
  2.   b: { 
  3.     c: 1, 
  4.     d: 2, 
  5.   } 
  6. }; 
  7.  
  8. console.log(a?.b?.c) 

会被转换成:

  1. var _a$b; 
  2.  
  3. const a = { 
  4.   b: { 
  5.     c: 1, 
  6.     d: 2 
  7.   } 
  8. }; 
  9. console.log( 
  10.   a === null || a === void 0 
  11.     ? void 0  
  12.     : (_a$b = a.b) === null || _a$b === void 0 
  13.       ? void 0 
  14.       : _a$b.c 
  15. ); 

 

Heny发布的相关些资料

Heny 目前是 babel 项目的负责人,之前发过一片介绍当前 babel 困境的文章: 知名团开源项目存活有多难?被数百万人使用的 Babel 陷入财务困境

上图推文链接:https://twitter.com/left_pad/status/1136282005170262016

感兴趣的可以去看看。

总结

?. 使用起来是非常方便的, 但如果用的不好, 也会隐藏本应该暴露出来的问题。

所以, 使用的时候一定要清楚自己在做什么。

?. 还有个小兄弟叫 空值合并运算符(??), 这里就不说了, 去 MDN 看文档吧。

今天就这么多, 希望对大家有所启发。

 

责任编辑:姜华 来源: 前端皮小蛋
相关推荐

2021-10-31 18:59:55

Python操作符用法

2010-07-14 14:55:07

Perl操作符

2009-08-19 17:26:28

C# 操作符

2011-02-25 09:36:06

java操作符

2011-04-08 16:26:14

JavaScript

2010-07-14 14:30:31

Perl操作符

2010-07-19 11:00:24

Perl操作符

2009-07-21 09:31:00

Scala操作符

2012-02-06 09:13:23

LINQ

2010-07-14 14:18:51

Perl操作符

2009-09-15 17:16:58

LINQ查询操作符

2009-09-16 09:09:23

Linq Contai

2010-01-28 11:16:28

C++操作符

2016-12-28 09:54:50

AndroidRxJava操作符

2009-11-30 16:48:08

PHP操作符

2009-07-14 18:34:22

Jython操作符重载

2017-01-03 16:12:13

RxJava操作符Android

2010-07-13 11:11:39

Perl标量

2022-10-08 07:49:55

New操作符函数

2009-11-27 09:41:56

LINQ
点赞
收藏

51CTO技术栈公众号