همونطور که احتمالا خیلی‌هاتون بامن هم‌عقیده هستین، جاوا اسکریپت زبان در دسترس، جالب و گاهی اوقات فوق‌العاده‌ایه. اما بعضی چیزا توی جاوا اسکریپت هست که واقعا عجیبه و غریبه! یکی از اون چیزا، وقتیه که سعی میکنیم یه آبجک نستد شده رو بگیریم و یهو این خطا به ما برمیگرده:

Cannot read property ‘foo’ of undefined

خیلی وقتایی که داریم در پروژه‌ جاوا اسکریپت کد میزینم، لازمه که از نستد آبجکت‌ها استفاده کرده و دیتا رو بطور امن دریافت کنیم، نه اینکه ازین خطاهای مزخرف بده و یکساعت بگردی ببینی کجای کارو اشتباه رفتی و چه مشکلی بوجود اومده.

میخوایم ببینیم برای حل این مشکل چه کارایی میشه انجام داد، بیایید با یه مثال بریم جلوو، فرض کنین همچین آبجکتی داریم:

const user = {
    id: 101,
    email: 'sadra@dev.com',
    personalInfo: {
        name: 'Sadra',
        address: {
            line1: 'kakh square',
            line2: '5 azar st',
            city: 'gorgan',
            state: 'GOLESTAN'
        }
    }
}

برای دسترسی به اسم یا شهر userمون اینجوری مینویسیم:

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

خب این ساده و مستقیمه.

اما یه موقع‌هایی اصلا آبجکت user فیلدی به نام personalInfo نداره. (الالخصوص وقتی داری با دیتابیس‌های non-relational مثل mongo کار میکنی). اونوق آبجکتت این شکلی میشه:

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

خب حالا وقتی تلاش میکنی که به مقدار name دسترسی پیدا کنی، با خطای Cannot read property ‘name’ of undefined مواجه میشی.

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

این برای اینه که سعی میکنیم به آبجکت name دسترسی پیدا کنیم، درحالی که حتی وجود خارجی نداره!

معمول‌ترین راه برای حل این مشکل که خیلی‌ها هم استفاده میکنن اینه:

const name = user && user.personalInfo ? user.personalInfo.name : null;
// undefined error will NOT be thrown as we check for existence before access

خب اینم یه راهه، ولی تا موقعی که آبجکتت مثل چیزی که بالا مثال زدیم ساده باشه. اگه یه آبجکت تودرتو تا ۵ ۶ مرحله داشته باشیم، اونموقع دستور شرطی که مینوسیم یه چیز احمقانه و عجیب قریبی مثل این میشه:

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;

با این notationای که استفاده کردیم، عمرا دیگه به خطای Cannot read property ‘name’ of undefined بربخوریم 😬.

اگه بخوام روش کنم که دقیقا چیکار کردیم، باید بگم داریم چک میکنیم اگر آبجکت user وجود نداره یه آبجکت خالی همینجوری رو هوا بساز و اگه وجود داشت، مرحله مرحله جلو میریم تا به property مورد نظرمون برسیم. و این باعث میشه که تاهیچوقتِ تاریخ، به خطایِ نبودنِ شیِ موردِ نظر بر نخوریم!

اما خب، متاسفانه با این روش نمیتونین به nester array object دسترسی داشته باشین!

Screaming from js

استفاده از Array Reduce برای دسترسی به آبجکتی از آرایه‌ی نستد شده!

متد Array Reduce یه راه قدرتمند و امن برای دسترسی به همچین مواردی هست. به مثال توجه کنید.

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

// pass in your object structure as array elements
const name = getNestedObject(user, ['personalInfo', 'name']);

// to access nested array, just pass in array index as an element the path array.
const city = getNestedObject(user, ['personalInfo', 'addresses', 0, 'city']);
// this will return the city from the first address item.

کتابخونه Typy

و در نهایت، این کتابخونه میتونه حلال همه مشکلاتتون باشه.

این کتابخونه علاوه به اینکه بهتون کمک میکنه تا به راحتی و با خیال راحت به هر آبجکت نستدی دسترسی پیدا کنید، بلکه کلی امکانات دیگه هم در ختیارتن میگذاره. این ابزار کاربردی از طریق کتابخونه npm در دسترس شماست - 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.

اما در پروژه‌‌هایی که کدهاتون لازمه در سبک‌ترین حالت ممکنن قرار بگیره (مثلا پروژه‌های front-end) و الالخصوص جاهایی که به یکی دو تا ازین متد‌ها بیشتر احتیاج ندارین، سعی کنین شخصا کد و متد موردنیازتون رو بنویسین و خودتون رو همیشه از شر دیپندنسی‌هاو لایبررهای عجیب غریب در امان نگهدارید 😉.

اگه سوال یا موردی بود، مثل همیشه در توییتر یا در تلگرام میتونیم باهم در ارتباط باشیم. 😉🍷