一天一个javascript小技巧【可选链操作符】

举报
江咏之 发表于 2022/07/06 00:02:46 2022/07/06
【摘要】 可选链操作符 尝试一下 可选链操作符 ( ?. ) 允许读取位于连接对象链深处的属性的值,而不必明确验证链中的每个引用是否有效。?. 操作符的功...

可选链操作符 ( ?. ) 允许读取位于连接对象链深处的属性的值,而不必明确验证链中的每个引用是否有效。?. 操作符的功能类似于 . 链式操作符,不同之处在于,在引用为空 (nullish ) (null 或者 undefined) 的情况下不会引起错误,该表达式短路返回值是 undefined。与函数调用一起使用时,如果给定的函数不存在,则返回 undefined

当尝试访问可能不存在的对象属性时,可选链操作符将会使表达式更短、更简明。在探索一个对象的内容时,如果不能确定哪些属性必定存在,可选链操作符也是很有帮助的。

语法

obj?.prop
obj?.[expr]
arr?.[index]
func?.(args)

描述

通过连接的对象的引用或函数可能是 undefinednull 时,可选链操作符提供了一种方法来简化被连接对象的值访问。

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

let nestedProp = obj.first && obj.first.second;
   
  
    Copy to Clipboard

    为了避免报错,在访问obj.first.second之前,要保证 obj.first 的值既不是 null,也不是 undefined。如果只是直接访问 obj.first.second,而不对 obj.first 进行校验,则有可能抛出错误。

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

    let nestedProp = obj.first?.second;
       
      
      Copy to Clipboard

      通过使用 ?. 操作符取代 . 操作符,JavaScript 会在尝试访问 obj.first.second 之前,先隐式地检查并确定 obj.first 既不是 null 也不是 undefined。如果obj.first null 或者 undefined,表达式将会短路计算直接返回 undefined

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

      let temp = obj.first;
      let nestedProp = ((temp === null || temp === undefined) ? undefined : temp.second);
         
        
      • 1
      Copy to Clipboard

      可选链与函数调用

      当尝试调用一个可能不存在的方法时也可以使用可选链。这将是很有帮助的,比如,当使用一个 API 的方法可能不可用时,要么因为实现的版本问题要么因为当前用户的设备不支持该功能。

      函数调用时如果被调用的方法不存在,使用可选链可以使表达式自动返回undefined而不是抛出一个异常。

      let result = someInterface.customMethod?.();
         
        
        Copy to Clipboard

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

        备注:如果 someInterface 自身是 null 或者 undefined ,异常 TypeError 仍会被抛出 someInterface is null 如果你希望允许 someInterface 也为 null 或者 undefined ,那么你需要像这样写 someInterface?.customMethod?.()

        处理可选的回调函数或者事件处理器

        如果使用解构赋值来解构的一个对象的回调函数或 fetch 方法,你可能得到不能当做函数直接调用的不存在的值,除非你已经校验了他们的存在性。使用?.的你可以忽略这些额外的校验:

        //  ES2019 的写法
        function doSomething(onContent, onError) {
          try {
            // ... do something with the data
          }
          catch (err) {
            if (onError) { // 校验 onError 是否真的存在
              onError(err.message);
            }
          }
        }
        
           
          
        • 1
        • 2
        • 3
        • 4
        • 5
        • 6
        • 7
        • 8
        • 9
        • 10
        • 11
        Copy to Clipboard
        // 使用可选链进行函数调用
        function doSomething(onContent, onError) {
          try {
           // ... do something with the data
          }
          catch (err) {
            onError?.(err.message); // 如果 onError 是 undefined 也不会有异常
          }
        }
        
           
          
        • 1
        • 2
        • 3
        • 4
        • 5
        • 6
        • 7
        • 8
        • 9
        Copy to Clipboard

        可选链和表达式

        当使用方括号与属性名的形式来访问属性时,你也可以使用可选链操作符:

        let nestedProp = obj?.['prop' + 'Name'];
           
          
          Copy to Clipboard

          可选链不能用于赋值

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

          可选链访问数组元素

          let arrayItem = arr?.[42];

          例子

          基本例子

          如下的例子在一个不含 bar 成员的 Map 中查找 bar 成员的 name 属性,因此结果是 undefined

          let myMap = new Map();
          myMap.set("foo", {name: "baz", desc: "inga"});
          
             
            
          • 1
          • 2

          let nameBar = myMap.get(“bar”)?.name;Copy to Clipboard

          短路计算

          当在表达式中使用可选链时,如果左操作数是 nullundefined,表达式将不会被计算,例如:

          let potentiallyNullObj = null;
          let x = 0;
          let prop = potentiallyNullObj?.[x++];
          
             
            
          • 1
          • 2
          • 3

          console.log(x); // x 将不会被递增,依旧输出 0
          Copy to Clipboard

          连用可选链操作符

          可以连续使用可选链读取多层嵌套结构:

          let customer = {
            name: "Carl",
            details: {
              age: 82,
              location: "Paradise Falls" // details 的 address 属性未有定义
            }
          };
          let customerCity = customer.details?.address?.city;
          
             
            
          • 1
          • 2
          • 3
          • 4
          • 5
          • 6
          • 7
          • 8

          // … 可选链也可以和函数调用一起使用
          let duration = vacations.trip?.getTime?.();
          Copy to Clipboard

          使用空值合并操作符

          空值合并操作符可以在使用可选链时设置一个默认值:

          let customer = {
            name: "Carl",
            details: { age: 82 }
          };
          let customerCity = customer?.city ?? "暗之城";
          console.log(customerCity); // “暗之城”
             
            
          • 1
          • 2
          • 3
          • 4
          • 5
          Copy to Clipboard

          规范

          规范 状态 备注
          ECMAScript (ECMA-262)
          optional expression
          Stage 4

          文章来源: jiangwenxin.blog.csdn.net,作者:前端江太公,版权归原作者所有,如需转载,请联系作者。

          原文链接:jiangwenxin.blog.csdn.net/article/details/125600394

          【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
          • 点赞
          • 收藏
          • 关注作者

          评论(0

          0/1000
          抱歉,系统识别当前为高风险访问,暂不支持该操作

          全部回复

          上滑加载中

          设置昵称

          在此一键设置昵称,即可参与社区互动!

          *长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

          *长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

          举报
          请填写举报理由
          0/200