在JavaScript中如何访问暂未存在的嵌套对象


Posted in Javascript onJune 18, 2019

前言

JavaScript 是个很神奇的东西。但是 JavaScript中的一些东西确实很奇怪,让人摸不着头脑。其中之一就是当你试图访问嵌套对象时,会遇到这个错误

Cannot read property 'foo' of undefined

在大多数情况下,处理嵌套的对象,通常我们需要安全地访问最内层嵌套的值。 来个粟子:

const user = {
  id: 101,
  email: 'jack@dev.com',
  personalInfo: {
    name: 'Jack',
    address: {
      line1: 'westwish st',
      line2: 'washmasher',
      city: 'wallas',
      state: 'WX'
    }
  }
}

当我们要访问user里面的name及city时,我们会这样写。

const name = user.personalInfo.name;
const userCity = user.personalInfo.address.city;

这是简单而直接的。

但是,由于某种原因,user 中的 personal不可用,对象结构将是这样的:

const user = {
  id: 101,
  email: 'jack@dev.com'
}

现在,如果你在试着访问 name ,将会得到一个 Cannot read property 'name' of undefined 的错误。

const name = user.personalInfo.name; // Cannot read property 'name' of undefined

这是因为我们试图访问对象中不在的 key 为 name 的属性。

大多数开发人员处理这种情况的常用方法如下,

const name = user && user.personalInfo ? user.personalInfo.name : null;

如果你的嵌套结构很简单,这是可以的,但是如果数据嵌套五或六层深,那么你的代码就会看起很混乱:

let city;
if (
  data && data.user && data.user.personalInfo &&
  data.user.personalInfo.addressDetails &&
  data.user.personalInfo.addressDetails.primaryAddress
  ) {
  city = data.user.personalInfo.addressDetails.primaryAddress;
}

有一些技巧可以处理这种混乱的对象结构。

Oliver Steele的嵌套对象访问模式

这是我个人的最爱,因为它使代码看起来干净简单。 我从 stackoverflow 中选择了这种风格,一旦你理解它是如何工作的,它就非常吸引人了。

const name = ((user || {}).personalInfo || {}).name;

使用这种表示法,永远不会遇到无法读取未定义的属性“name”。做法是检查用户是否存在,如果不存在,就创建一个空对象,这样,下一个级别的键将始终从存在的对象访问。

不幸的是,你不能使用此技巧访问嵌套数组。

使用数组Reduce访问嵌套对象

Array reduce 方法非常强大,可用于安全地访问嵌套对象。

const getNestedObject = (nestedObj, pathArr) => {
  return pathArr.reduce((obj, key) =>
    (obj && obj[key] !== 'undefined') ? obj[key] : null, nestedObj);
}

// 将对象结构作为数组元素传入
const name = getNestedObject(user, ['personalInfo', 'name']);

// 要访问嵌套数组,只需将数组索引作为数组元素传入。.
const city = getNestedObject(user, ['personalInfo', 'addresses', 0, 'city']);
// 这将从 addresses 中的第一层返回 city

Typy

如果你认为上面的方法太过非主流,那么可以使用 Typy库。除了安全访问嵌套对象之外,它还可以做很多很棒的事情。

如果使用Typy,代码将如下所示

import t from 'typy';

const name = t(user, 'personalInfo.name').safeObject;
const city = t(user, 'personalInfo.addresses[0].city').safeObject;
// address is an array

这里还有一些其他的库,如 Lodash 和 Ramda,可以做到这一点。但是在轻量级前端项目中,特别是如果你只需要这些库中的一两个方法时,最好选择另一个轻量级库,或者编写自己的库。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对三水点靠木的支持。

Javascript 相关文章推荐
『jQuery』名称冲突使用noConflict方法解决
Apr 22 Javascript
JS实现点击下载的小例子
Jul 10 Javascript
深入理解JavaScript系列(33):设计模式之策略模式详解
Mar 03 Javascript
使用Node.js配合Nginx实现高负载网络
Jun 28 Javascript
jquery实现适用于门户站的导航下拉菜单效果代码
Aug 24 Javascript
移动手机APP手指滑动切换图片特效附源码下载
Nov 30 Javascript
vue-router单页面路由
Jun 17 Javascript
JS基于对象的特性实现去除数组中重复项功能详解
Nov 17 Javascript
Vue.js分页组件实现:diVuePagination的使用详解
Jan 10 Javascript
webpack里使用jquery.mCustomScrollbar插件的方法
May 30 jQuery
Node.js实现简单的爬取的示例代码
Jun 25 Javascript
解决Vue watch里调用方法的坑
Nov 07 Javascript
用Vue.js在浏览器中实现裁剪图像功能
Jun 18 #Javascript
JS删除String里某个字符的方法
Jan 06 #Javascript
简单了解Javscript中兄弟ifream的方法调用
Jun 17 #Javascript
vue中typescript装饰器的使用方法超实用教程
Jun 17 #Javascript
简单学习5种处理Vue.js异常的方法
Jun 17 #Javascript
js/jQuery实现全选效果
Jun 17 #jQuery
解决微信浏览器缓存站点入口文件(IIS部署Vue项目)
Jun 17 #Javascript
You might like
PHP 第三节 变量介绍
2012/04/28 PHP
关于Sphinx创建全文检索的索引介绍
2013/06/25 PHP
destoon二次开发常用数据库操作
2014/06/21 PHP
JavaScript使用过程中需要注意的地方和一些基本语法
2010/08/26 Javascript
js自动下载文件到本地的实现代码
2013/04/28 Javascript
用js实现in_array的方法
2013/11/05 Javascript
node.js中的path.isAbsolute方法使用说明
2014/12/08 Javascript
BootStrap点击下拉菜单项后显示一个新的输入框实现代码
2016/05/16 Javascript
使用jquery获取url及url参数的简单实例
2016/06/14 Javascript
基于vuejs+webpack的日期选择插件
2020/05/21 Javascript
vue组件Prop传递数据的实现示例
2017/08/17 Javascript
Bootstrap Table 删除和批量删除
2017/09/22 Javascript
React注册倒计时功能的实现
2018/09/06 Javascript
js指定日期增加指定月份的实现方法
2018/12/19 Javascript
Node.js系列之连接DB的方法(3)
2019/08/30 Javascript
Vue extend的基本用法(实例详解)
2019/12/09 Javascript
Python判断列表是否已排序的各种方法及其性能分析
2016/06/20 Python
浅谈python数据类型及类型转换
2017/12/18 Python
python实现扫描局域网指定网段ip的方法
2019/04/16 Python
Django REST framework 视图和路由详解
2019/07/19 Python
Python使用type关键字创建类步骤详解
2019/07/23 Python
Django的Modelforms用法简介
2019/07/27 Python
Python实现语音识别和语音合成功能
2019/09/20 Python
Selenium 滚动页面至元素可见的方法
2020/03/18 Python
Python Flask框架实现简单加法工具过程解析
2020/06/03 Python
python中str内置函数用法总结
2020/12/27 Python
详解如何在css3打包后自动追加前缀插件:autoprefixer
2018/12/18 HTML / CSS
什么是触发器(trigger)? 触发器有什么作用?
2013/09/18 面试题
介绍一下如何优化MySql
2016/12/20 面试题
学校门卫工作职责
2013/12/07 职场文书
党员的自我评价范文
2014/01/02 职场文书
安全生产汇报材料
2014/02/17 职场文书
南京市纪委监察局整改方案
2014/09/16 职场文书
员工开除通知书
2015/04/25 职场文书
2016全国“质量月”活动标语口号
2015/12/26 职场文书
《金钱的魔力》教学反思
2016/02/20 职场文书