<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0">
    <channel>
        <title>پست‌های انتشارات آموزش‌های فارسی جاوا اسکریپت</title>
        <link>https://virgool.io/js-tuts/feed</link>
        <description>آموزش‌های مدرن جاوا اسکریپت از پایه تا پیشرفته، به همراه توضیحات عمیق.</description>
        <language>fa</language>
        <pubDate>2026-06-16 22:38:41</pubDate>
        <image>
            <url>https://files.virgool.io/upload/publication/zzfiabpymeyr/1xfoth.png</url>
            <title>آموزش‌های فارسی جاوا اسکریپت</title>
            <link>https://virgool.io/js-tuts</link>
        </image>

                    <item>
                <title>زنجیره اختیاری یا Optional Chaining در جاوااسکریپت</title>
                <link>https://virgool.io/js-tuts/%D8%B2%D9%86%D8%AC%DB%8C%D8%B1%D9%87-%D8%A7%D8%AE%D8%AA%DB%8C%D8%A7%D8%B1%DB%8C-%DB%8C%D8%A7-optional-chaining-%D8%AF%D8%B1-%D8%AC%D8%A7%D9%88%D8%A7%D8%A7%D8%B3%DA%A9%D8%B1%DB%8C%D9%BE%D8%AA-bnld7nngxg9c</link>
                <description>زنجیره ای اختیاری روشی بدون خطا برای دستیابی به ویژگی های(properties) داخلی شی است حتی در زمانی که ویژگی میانی وجود نداشته باشدمشکل ویژگی‌های (property) که وجود ندارنداگر به تازگی شروع به خواندن آموزش و یادگیری جاوا اسکریپت کرده اید ، شاید این مشکل را هنوز لمس نکرده اید ، اما این یک مشکل کاملاً رایج است.برای مثال, بیاید یک شی(object) برای ذخیره کردن اطلاعات کاربرانمان در نظر بگیریم.  اکثر کاربران ما ویژگی(property) آدرس را در user.address و خیابان  را در user.address.street را دارند ولی بعضی از انان این اطلاعات را ارائه نکرده اند.در همچین مثالی تلاش ما برای دریافت مقدار  user.address.street, برای کاربری که آدرس ندارد با خطا مواجه شود :let user = {}; // یک کاربر بدون ویژگی &amp;quotaddress&amp;quotalert&#40;user.address.street&#41;; // خطااین یک خروجی قابل حدس است٬ جاوااسکریپت اینگونه کار میکند٬تا زمانی که user.address  برابر با undefined است تلاش برای گرفتن آن با خطا مواجه میشود. ولی  در بسیاری از موارد عملی ، ما ترجیح می دهیم به جای خطا ، ‍undefined  را دریافت کنیم (به معنای &quot;بدون خیابان&quot;).یا مثالی دیگر در توسعه وب٬ ما میخواهیم اطلاعاتی در مورد اِلمانی در صفحه را بگیریم.این اِلمان  توسط این عبارت document.querySelector(&#x27;.elem&#x27;) گرفته میشود که ممکن است گاهی وجود نداشته باشد:document.querySelector(&#039;.elem&#039;) //  خواهد شد اگر المنت وجود نداشته باشد null
let html = document.querySelector(&#039;.elem&#039;);  // اگر باشد خطا خواهد داد NULLیکبار دیگر٬ اگر اِلمان وجود نداشته باشد ما با مقدار  NULL نمیتوانیم به   دسترسی داشته باشیم. و در بعضی موارد وقتی که نبود اِلمان طبیعی است ، ما می خواهیم از خطا جلوگیری کنیم و فقط &quot;html = null&quot; را قبول کنیم.چگونه میتوانیم از این استفاده کنیم ؟راه‌حل روشن این است که مقدار آن را با if یا عمگر شرطی ? بررسی کنیم قبل از اینکه به ویژگی (property) آن دسترسی پیدا کنیمlet user = {};
alert&#40;user.address ? user.address.street : undefined&#41;;الان بدون خطا کار میکند... ولی اصلا زیبا نیست. همانطور که میبینید مقدار&quot;user.address&quot;  دوبار در کد تکرار شده است. برای دسترسی به ویژگی‌های(property) با تو در تویی زیاد نیاز به تکرار بیشتری لازم است و این مشکل ایجاد میکند.برای مثال بیاید مقدار عبارت  user.address.street.nameرا بگیریم.در این صورت ما باید هم  user.address  و    user.address.street را بررسی کنیم :let user = {}; // کاربر بدون آدرس
alert&#40;user.address ? user.address.street ? user.address.street.name : null : null&#41;;این افتضاح است یک نفر ممکن حتی با درک این کد مشکل داشته باشد.قبل از اینکه زنجیره اختیاری به جاوااسکریپت اضافه شود مردم از عملگر &amp;&amp; برای بعضی مواقع استفاده میکردند:نگران نباشید :) ٬ راه های بهتری  هم هست میتوانیم از عملگر &amp;&amp; استفاده کنیم.let user = {}; // کاربر بدون آدرس
alert&#40; user.address &amp;&amp; user.address.street &amp;&amp; user.address.street.name &#41;; // undefined (بدون خطا) اند(AND) کردن کل مسیر رسیدن به ویژگی ، وجود همه ویژگی ها(property) را تضمین می کند(اگر ارزیابی متوقف نشود) ، اما نوشتن آن دست و پا گیر است.همانطور که میبنید نام ویژگی ها همچنان در کد تکرار میشوند. به طور مثال در قطعه کد بالا user.address  سه بار تکرار شده است.و حالا در نهایت زنجیره اختیاری آمده است که ما را نجات دهد!زنجیره اختیاریزنجیره ای اختیاری ?. بررسی را متوقف میکند  اگر مقدار قبل از قسمت  ?.  برابر با undefined یا null باشد و مقدار undefined را برمیگرداند.یا به عبارت دیگر  value?.prop :برابر است با value.prop اگر value‍ وجود داشته باشددر غیر اینصورت (زمانی که value برابر با undefined/null است) مقدار undefined را برمیگرداند..? این یک دسترسی مطمئن به user.address.street  است: let user = {}; // کاربر بدون آدرس
alert&#40; user?.address?.street &#41;; // undefined (بدون خطا)حالا کد  خیلی کوتاه‌تر و تمیزتر است و بدون هیچ تکرار اضافه‌ای :)خواندن ویژگی(property) آدرس با  user?.address کار خواهد کرد حتی زمانی هم که  شی(آبجکت) user وجود ندارد :let user = null;
alert&#40; user?.address &#41;; // undefined
alert&#40; user?.address.street &#41;; // undefinedلطفا توجه داشته باشید : سینتکس ?. مقدارهای قبلی را اختیاری میکند نه مقدارهای جلوی آن را.در مثال بالا user?.  به user مقدار null/undefined خواهد داد.در مثال بالا  user?.address.street  فقط به  user‍  اجازه میدهد که null/undefined باشد. مثلا در این کد user?.address.street.name  عبارت ‍.? اجازه میدهد که user برابر با null/undefined  باشد. این همه کاری است که انجام میدهد. ویژگی های جلویی به سبک معمولی به ویژگی ها دسترسی دارند.اگر ما میخواهیم بعضی از ویژگی ها را اختیاری کنیم میتوانیم تعداد بیشتری از . را با .? جایگزین کنیماز طرف دیگر ، اگر ‍‍user وجود داشته باشد ، پس باید ویژگی user.address داشته باشد ، در غیر این صورت user؟.address.streetدر نقطه دوم خطا می دهد.ما باید از &#x60;?.&#x60; فقط زمانی استفاده کنیم که عدم وجود چیزی اشکالی ندارد.برای مثال اگر طبق منطق و لاجیک ما باید شی(object)&#x60;user&#x60; وجود داشته باشد ولی address اختیاری است. پس ما باید اینگونه بنویسیم user.address?.street نه user?.address?.streetبنابراین ، اگر تصادفاً به دلیل اشتباهی ‍user  برابر با undefined باشد، شاهد یک خطای برنامه نویسی در مورد آن خواهیم بود و آن را برطرف خواهیم کرد.برای زنجیره اختیاری باید متغیر حتما تعریف شده باشد ( let/const/var user یا توابع  ). زنجیره ای اختیاری فقط برای متغیرهای تعریف شده کار می کند.// ReferenceError: user is not defined
user?.address;از زنجیره اختیاری میتوان برای صدا زدن توابع هم استفاده کرد :let userAdmin = {
 admin() {
 alert&#40;&amp;quotI am admin&amp;quot&#41;;
}
};
let userGuest = {};
userAdmin.admin?.(); // I am admin
userGuest.admin?.(); // هیچی (هیچ متدی نیست)اگر ما میخواهیم از براکت به جای نقطه برای دسترسی به ویژگی(property) استفاده کنیم زنجیره اختیاری برای آن حالت هم کارایی دارد.let user1 = {
firstName: &amp;quotJohn&amp;quot
};

let user2 = null;

let key = &amp;quotfirstName&amp;quot

alert&#40; user1?.[key] &#41;; // John
alert&#40; user2?.[key] &#41;; // undefined
alert&#40; user1?.[key]?.something?.not?.existing&#41;; // undefinedخلاصهسینتکس ?.  سه حالت دارد:حالت اول: obj?.prop -  مقدار ‍‍obj.prop را برمیگرداند اگر obj وجود داشته باشد در غیر اینصورت مقدار undefined  را برمیگرداندحالت دوم: [obj?.[prop -  مقدار ‍‍[obj.[prop را برمیگرداند اگر obj وجود داشته باشد در غیر اینصورت مقدار undefined  را برمیگرداندحالت سوم: ()obj.method()  -   ‍‍obj?.method را صدا میزند اگر obj وجود داشته باشد در غیر اینصورت مقدار undefined  را برمیگرداندهمانطور که می بینیم ، همه آنها ساده و آسان برای استفاده هستند. ?.  سمت چپ را از نظر null/undefined بررسی می کند و اجازه می دهد تا ارزیابی ادامه یابد اگر برابر باnull/undefined  نباشد.زنجیر ?. امکان دسترسی به خواص تودرتو را هم فراهم میکند.با این حال هنوز ما باید ?. را با دقت اعمال کنیم ، فقط درصورتی قابل قبول است که سمت چپ ممکن است وجود نداشته باشد.با این حال خطاهای برنامه نویسی را از ما مخفی نمیکند اگر آنها اتفاق بیافتند.اگر از جاوا اسکریپت کارهای حرفه ای باشید، یا برای حرفه ای شدن در این زمینه تلاش کرده باشید، حتما سایت javascript.info رو میشناسیداین سایت آموزشهای بسیار کامل و عمیق از زبان جاوا اسکریپت رو به صورت متنی و متن باز منتشر میکنه.این آموزش ها توی اکانت سازمانی javascript.info در گیتهاب وجود داره و تا به این لحظه به حدود ۵۰ زبان دنیا ترجمه شده (یا در حال ترجمه هست).وقتی ترجمه ی یک زبان به حد نصابی میرسه، ساب دومینی برای اون زبان در javascript.info ساخته میشه. برای نمونه tr.javascript.info در حال حاضر این آموزش ها رو به زبان ترکی نمایش میده.خوشبختانه با تلاش تعدادی از علاقه مندان به کارهای متن باز بخشی از این آموزش ها به فارسی هم ترجمه شده(این مقاله هم ترجمه یکی از قسمت هاست) و میتونید برای یادگیری و یا ترجمه ، به ریپازیتوری رسمی زبان فارسی در آدرس پایین سر بزنید و شماهم مشارکت(contribute) کنید: لینکبه امید روزی که fa.javascript.info رو ببینیم ?پ.ن: گروه تلگرامی ای برای ارتباط بهتر javascript کارها ایجاد کردیم :t.me/jsforeveryone</description>
                <category>آموزش‌های فارسی جاوا اسکریپت</category>
                <author>مهدی مومنی</author>
                <pubDate>Fri, 30 Oct 2020 20:40:33 +0330</pubDate>
            </item>
            </channel>
</rss>